##// END OF EJS Templates
revlog-compression: fix computation of engine availability...
marmoute -
r47610:3aa78f2a default
parent child Browse files
Show More
@@ -1,3752 +1,3754 b''
1 # localrepo.py - read/write repository class for mercurial
1 # localrepo.py - read/write repository class for mercurial
2 #
2 #
3 # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com>
3 # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 from __future__ import absolute_import
8 from __future__ import absolute_import
9
9
10 import errno
10 import errno
11 import functools
11 import functools
12 import os
12 import os
13 import random
13 import random
14 import sys
14 import sys
15 import time
15 import time
16 import weakref
16 import weakref
17
17
18 from .i18n import _
18 from .i18n import _
19 from .node import (
19 from .node import (
20 bin,
20 bin,
21 hex,
21 hex,
22 nullid,
22 nullid,
23 nullrev,
23 nullrev,
24 sha1nodeconstants,
24 sha1nodeconstants,
25 short,
25 short,
26 )
26 )
27 from .pycompat import (
27 from .pycompat import (
28 delattr,
28 delattr,
29 getattr,
29 getattr,
30 )
30 )
31 from . import (
31 from . import (
32 bookmarks,
32 bookmarks,
33 branchmap,
33 branchmap,
34 bundle2,
34 bundle2,
35 bundlecaches,
35 bundlecaches,
36 changegroup,
36 changegroup,
37 color,
37 color,
38 commit,
38 commit,
39 context,
39 context,
40 dirstate,
40 dirstate,
41 dirstateguard,
41 dirstateguard,
42 discovery,
42 discovery,
43 encoding,
43 encoding,
44 error,
44 error,
45 exchange,
45 exchange,
46 extensions,
46 extensions,
47 filelog,
47 filelog,
48 hook,
48 hook,
49 lock as lockmod,
49 lock as lockmod,
50 match as matchmod,
50 match as matchmod,
51 mergestate as mergestatemod,
51 mergestate as mergestatemod,
52 mergeutil,
52 mergeutil,
53 metadata as metadatamod,
53 metadata as metadatamod,
54 namespaces,
54 namespaces,
55 narrowspec,
55 narrowspec,
56 obsolete,
56 obsolete,
57 pathutil,
57 pathutil,
58 phases,
58 phases,
59 pushkey,
59 pushkey,
60 pycompat,
60 pycompat,
61 rcutil,
61 rcutil,
62 repoview,
62 repoview,
63 requirements as requirementsmod,
63 requirements as requirementsmod,
64 revlog,
64 revlog,
65 revset,
65 revset,
66 revsetlang,
66 revsetlang,
67 scmutil,
67 scmutil,
68 sparse,
68 sparse,
69 store as storemod,
69 store as storemod,
70 subrepoutil,
70 subrepoutil,
71 tags as tagsmod,
71 tags as tagsmod,
72 transaction,
72 transaction,
73 txnutil,
73 txnutil,
74 util,
74 util,
75 vfs as vfsmod,
75 vfs as vfsmod,
76 )
76 )
77
77
78 from .interfaces import (
78 from .interfaces import (
79 repository,
79 repository,
80 util as interfaceutil,
80 util as interfaceutil,
81 )
81 )
82
82
83 from .utils import (
83 from .utils import (
84 hashutil,
84 hashutil,
85 procutil,
85 procutil,
86 stringutil,
86 stringutil,
87 )
87 )
88
88
89 from .revlogutils import (
89 from .revlogutils import (
90 concurrency_checker as revlogchecker,
90 concurrency_checker as revlogchecker,
91 constants as revlogconst,
91 constants as revlogconst,
92 )
92 )
93
93
94 release = lockmod.release
94 release = lockmod.release
95 urlerr = util.urlerr
95 urlerr = util.urlerr
96 urlreq = util.urlreq
96 urlreq = util.urlreq
97
97
98 # set of (path, vfs-location) tuples. vfs-location is:
98 # set of (path, vfs-location) tuples. vfs-location is:
99 # - 'plain for vfs relative paths
99 # - 'plain for vfs relative paths
100 # - '' for svfs relative paths
100 # - '' for svfs relative paths
101 _cachedfiles = set()
101 _cachedfiles = set()
102
102
103
103
104 class _basefilecache(scmutil.filecache):
104 class _basefilecache(scmutil.filecache):
105 """All filecache usage on repo are done for logic that should be unfiltered"""
105 """All filecache usage on repo are done for logic that should be unfiltered"""
106
106
107 def __get__(self, repo, type=None):
107 def __get__(self, repo, type=None):
108 if repo is None:
108 if repo is None:
109 return self
109 return self
110 # proxy to unfiltered __dict__ since filtered repo has no entry
110 # proxy to unfiltered __dict__ since filtered repo has no entry
111 unfi = repo.unfiltered()
111 unfi = repo.unfiltered()
112 try:
112 try:
113 return unfi.__dict__[self.sname]
113 return unfi.__dict__[self.sname]
114 except KeyError:
114 except KeyError:
115 pass
115 pass
116 return super(_basefilecache, self).__get__(unfi, type)
116 return super(_basefilecache, self).__get__(unfi, type)
117
117
118 def set(self, repo, value):
118 def set(self, repo, value):
119 return super(_basefilecache, self).set(repo.unfiltered(), value)
119 return super(_basefilecache, self).set(repo.unfiltered(), value)
120
120
121
121
122 class repofilecache(_basefilecache):
122 class repofilecache(_basefilecache):
123 """filecache for files in .hg but outside of .hg/store"""
123 """filecache for files in .hg but outside of .hg/store"""
124
124
125 def __init__(self, *paths):
125 def __init__(self, *paths):
126 super(repofilecache, self).__init__(*paths)
126 super(repofilecache, self).__init__(*paths)
127 for path in paths:
127 for path in paths:
128 _cachedfiles.add((path, b'plain'))
128 _cachedfiles.add((path, b'plain'))
129
129
130 def join(self, obj, fname):
130 def join(self, obj, fname):
131 return obj.vfs.join(fname)
131 return obj.vfs.join(fname)
132
132
133
133
134 class storecache(_basefilecache):
134 class storecache(_basefilecache):
135 """filecache for files in the store"""
135 """filecache for files in the store"""
136
136
137 def __init__(self, *paths):
137 def __init__(self, *paths):
138 super(storecache, self).__init__(*paths)
138 super(storecache, self).__init__(*paths)
139 for path in paths:
139 for path in paths:
140 _cachedfiles.add((path, b''))
140 _cachedfiles.add((path, b''))
141
141
142 def join(self, obj, fname):
142 def join(self, obj, fname):
143 return obj.sjoin(fname)
143 return obj.sjoin(fname)
144
144
145
145
146 class mixedrepostorecache(_basefilecache):
146 class mixedrepostorecache(_basefilecache):
147 """filecache for a mix files in .hg/store and outside"""
147 """filecache for a mix files in .hg/store and outside"""
148
148
149 def __init__(self, *pathsandlocations):
149 def __init__(self, *pathsandlocations):
150 # scmutil.filecache only uses the path for passing back into our
150 # scmutil.filecache only uses the path for passing back into our
151 # join(), so we can safely pass a list of paths and locations
151 # join(), so we can safely pass a list of paths and locations
152 super(mixedrepostorecache, self).__init__(*pathsandlocations)
152 super(mixedrepostorecache, self).__init__(*pathsandlocations)
153 _cachedfiles.update(pathsandlocations)
153 _cachedfiles.update(pathsandlocations)
154
154
155 def join(self, obj, fnameandlocation):
155 def join(self, obj, fnameandlocation):
156 fname, location = fnameandlocation
156 fname, location = fnameandlocation
157 if location == b'plain':
157 if location == b'plain':
158 return obj.vfs.join(fname)
158 return obj.vfs.join(fname)
159 else:
159 else:
160 if location != b'':
160 if location != b'':
161 raise error.ProgrammingError(
161 raise error.ProgrammingError(
162 b'unexpected location: %s' % location
162 b'unexpected location: %s' % location
163 )
163 )
164 return obj.sjoin(fname)
164 return obj.sjoin(fname)
165
165
166
166
167 def isfilecached(repo, name):
167 def isfilecached(repo, name):
168 """check if a repo has already cached "name" filecache-ed property
168 """check if a repo has already cached "name" filecache-ed property
169
169
170 This returns (cachedobj-or-None, iscached) tuple.
170 This returns (cachedobj-or-None, iscached) tuple.
171 """
171 """
172 cacheentry = repo.unfiltered()._filecache.get(name, None)
172 cacheentry = repo.unfiltered()._filecache.get(name, None)
173 if not cacheentry:
173 if not cacheentry:
174 return None, False
174 return None, False
175 return cacheentry.obj, True
175 return cacheentry.obj, True
176
176
177
177
178 class unfilteredpropertycache(util.propertycache):
178 class unfilteredpropertycache(util.propertycache):
179 """propertycache that apply to unfiltered repo only"""
179 """propertycache that apply to unfiltered repo only"""
180
180
181 def __get__(self, repo, type=None):
181 def __get__(self, repo, type=None):
182 unfi = repo.unfiltered()
182 unfi = repo.unfiltered()
183 if unfi is repo:
183 if unfi is repo:
184 return super(unfilteredpropertycache, self).__get__(unfi)
184 return super(unfilteredpropertycache, self).__get__(unfi)
185 return getattr(unfi, self.name)
185 return getattr(unfi, self.name)
186
186
187
187
188 class filteredpropertycache(util.propertycache):
188 class filteredpropertycache(util.propertycache):
189 """propertycache that must take filtering in account"""
189 """propertycache that must take filtering in account"""
190
190
191 def cachevalue(self, obj, value):
191 def cachevalue(self, obj, value):
192 object.__setattr__(obj, self.name, value)
192 object.__setattr__(obj, self.name, value)
193
193
194
194
195 def hasunfilteredcache(repo, name):
195 def hasunfilteredcache(repo, name):
196 """check if a repo has an unfilteredpropertycache value for <name>"""
196 """check if a repo has an unfilteredpropertycache value for <name>"""
197 return name in vars(repo.unfiltered())
197 return name in vars(repo.unfiltered())
198
198
199
199
200 def unfilteredmethod(orig):
200 def unfilteredmethod(orig):
201 """decorate method that always need to be run on unfiltered version"""
201 """decorate method that always need to be run on unfiltered version"""
202
202
203 @functools.wraps(orig)
203 @functools.wraps(orig)
204 def wrapper(repo, *args, **kwargs):
204 def wrapper(repo, *args, **kwargs):
205 return orig(repo.unfiltered(), *args, **kwargs)
205 return orig(repo.unfiltered(), *args, **kwargs)
206
206
207 return wrapper
207 return wrapper
208
208
209
209
210 moderncaps = {
210 moderncaps = {
211 b'lookup',
211 b'lookup',
212 b'branchmap',
212 b'branchmap',
213 b'pushkey',
213 b'pushkey',
214 b'known',
214 b'known',
215 b'getbundle',
215 b'getbundle',
216 b'unbundle',
216 b'unbundle',
217 }
217 }
218 legacycaps = moderncaps.union({b'changegroupsubset'})
218 legacycaps = moderncaps.union({b'changegroupsubset'})
219
219
220
220
221 @interfaceutil.implementer(repository.ipeercommandexecutor)
221 @interfaceutil.implementer(repository.ipeercommandexecutor)
222 class localcommandexecutor(object):
222 class localcommandexecutor(object):
223 def __init__(self, peer):
223 def __init__(self, peer):
224 self._peer = peer
224 self._peer = peer
225 self._sent = False
225 self._sent = False
226 self._closed = False
226 self._closed = False
227
227
228 def __enter__(self):
228 def __enter__(self):
229 return self
229 return self
230
230
231 def __exit__(self, exctype, excvalue, exctb):
231 def __exit__(self, exctype, excvalue, exctb):
232 self.close()
232 self.close()
233
233
234 def callcommand(self, command, args):
234 def callcommand(self, command, args):
235 if self._sent:
235 if self._sent:
236 raise error.ProgrammingError(
236 raise error.ProgrammingError(
237 b'callcommand() cannot be used after sendcommands()'
237 b'callcommand() cannot be used after sendcommands()'
238 )
238 )
239
239
240 if self._closed:
240 if self._closed:
241 raise error.ProgrammingError(
241 raise error.ProgrammingError(
242 b'callcommand() cannot be used after close()'
242 b'callcommand() cannot be used after close()'
243 )
243 )
244
244
245 # We don't need to support anything fancy. Just call the named
245 # We don't need to support anything fancy. Just call the named
246 # method on the peer and return a resolved future.
246 # method on the peer and return a resolved future.
247 fn = getattr(self._peer, pycompat.sysstr(command))
247 fn = getattr(self._peer, pycompat.sysstr(command))
248
248
249 f = pycompat.futures.Future()
249 f = pycompat.futures.Future()
250
250
251 try:
251 try:
252 result = fn(**pycompat.strkwargs(args))
252 result = fn(**pycompat.strkwargs(args))
253 except Exception:
253 except Exception:
254 pycompat.future_set_exception_info(f, sys.exc_info()[1:])
254 pycompat.future_set_exception_info(f, sys.exc_info()[1:])
255 else:
255 else:
256 f.set_result(result)
256 f.set_result(result)
257
257
258 return f
258 return f
259
259
260 def sendcommands(self):
260 def sendcommands(self):
261 self._sent = True
261 self._sent = True
262
262
263 def close(self):
263 def close(self):
264 self._closed = True
264 self._closed = True
265
265
266
266
267 @interfaceutil.implementer(repository.ipeercommands)
267 @interfaceutil.implementer(repository.ipeercommands)
268 class localpeer(repository.peer):
268 class localpeer(repository.peer):
269 '''peer for a local repo; reflects only the most recent API'''
269 '''peer for a local repo; reflects only the most recent API'''
270
270
271 def __init__(self, repo, caps=None):
271 def __init__(self, repo, caps=None):
272 super(localpeer, self).__init__()
272 super(localpeer, self).__init__()
273
273
274 if caps is None:
274 if caps is None:
275 caps = moderncaps.copy()
275 caps = moderncaps.copy()
276 self._repo = repo.filtered(b'served')
276 self._repo = repo.filtered(b'served')
277 self.ui = repo.ui
277 self.ui = repo.ui
278
278
279 if repo._wanted_sidedata:
279 if repo._wanted_sidedata:
280 formatted = bundle2.format_remote_wanted_sidedata(repo)
280 formatted = bundle2.format_remote_wanted_sidedata(repo)
281 caps.add(b'exp-wanted-sidedata=' + formatted)
281 caps.add(b'exp-wanted-sidedata=' + formatted)
282
282
283 self._caps = repo._restrictcapabilities(caps)
283 self._caps = repo._restrictcapabilities(caps)
284
284
285 # Begin of _basepeer interface.
285 # Begin of _basepeer interface.
286
286
287 def url(self):
287 def url(self):
288 return self._repo.url()
288 return self._repo.url()
289
289
290 def local(self):
290 def local(self):
291 return self._repo
291 return self._repo
292
292
293 def peer(self):
293 def peer(self):
294 return self
294 return self
295
295
296 def canpush(self):
296 def canpush(self):
297 return True
297 return True
298
298
299 def close(self):
299 def close(self):
300 self._repo.close()
300 self._repo.close()
301
301
302 # End of _basepeer interface.
302 # End of _basepeer interface.
303
303
304 # Begin of _basewirecommands interface.
304 # Begin of _basewirecommands interface.
305
305
306 def branchmap(self):
306 def branchmap(self):
307 return self._repo.branchmap()
307 return self._repo.branchmap()
308
308
309 def capabilities(self):
309 def capabilities(self):
310 return self._caps
310 return self._caps
311
311
312 def clonebundles(self):
312 def clonebundles(self):
313 return self._repo.tryread(bundlecaches.CB_MANIFEST_FILE)
313 return self._repo.tryread(bundlecaches.CB_MANIFEST_FILE)
314
314
315 def debugwireargs(self, one, two, three=None, four=None, five=None):
315 def debugwireargs(self, one, two, three=None, four=None, five=None):
316 """Used to test argument passing over the wire"""
316 """Used to test argument passing over the wire"""
317 return b"%s %s %s %s %s" % (
317 return b"%s %s %s %s %s" % (
318 one,
318 one,
319 two,
319 two,
320 pycompat.bytestr(three),
320 pycompat.bytestr(three),
321 pycompat.bytestr(four),
321 pycompat.bytestr(four),
322 pycompat.bytestr(five),
322 pycompat.bytestr(five),
323 )
323 )
324
324
325 def getbundle(
325 def getbundle(
326 self,
326 self,
327 source,
327 source,
328 heads=None,
328 heads=None,
329 common=None,
329 common=None,
330 bundlecaps=None,
330 bundlecaps=None,
331 remote_sidedata=None,
331 remote_sidedata=None,
332 **kwargs
332 **kwargs
333 ):
333 ):
334 chunks = exchange.getbundlechunks(
334 chunks = exchange.getbundlechunks(
335 self._repo,
335 self._repo,
336 source,
336 source,
337 heads=heads,
337 heads=heads,
338 common=common,
338 common=common,
339 bundlecaps=bundlecaps,
339 bundlecaps=bundlecaps,
340 remote_sidedata=remote_sidedata,
340 remote_sidedata=remote_sidedata,
341 **kwargs
341 **kwargs
342 )[1]
342 )[1]
343 cb = util.chunkbuffer(chunks)
343 cb = util.chunkbuffer(chunks)
344
344
345 if exchange.bundle2requested(bundlecaps):
345 if exchange.bundle2requested(bundlecaps):
346 # When requesting a bundle2, getbundle returns a stream to make the
346 # When requesting a bundle2, getbundle returns a stream to make the
347 # wire level function happier. We need to build a proper object
347 # wire level function happier. We need to build a proper object
348 # from it in local peer.
348 # from it in local peer.
349 return bundle2.getunbundler(self.ui, cb)
349 return bundle2.getunbundler(self.ui, cb)
350 else:
350 else:
351 return changegroup.getunbundler(b'01', cb, None)
351 return changegroup.getunbundler(b'01', cb, None)
352
352
353 def heads(self):
353 def heads(self):
354 return self._repo.heads()
354 return self._repo.heads()
355
355
356 def known(self, nodes):
356 def known(self, nodes):
357 return self._repo.known(nodes)
357 return self._repo.known(nodes)
358
358
359 def listkeys(self, namespace):
359 def listkeys(self, namespace):
360 return self._repo.listkeys(namespace)
360 return self._repo.listkeys(namespace)
361
361
362 def lookup(self, key):
362 def lookup(self, key):
363 return self._repo.lookup(key)
363 return self._repo.lookup(key)
364
364
365 def pushkey(self, namespace, key, old, new):
365 def pushkey(self, namespace, key, old, new):
366 return self._repo.pushkey(namespace, key, old, new)
366 return self._repo.pushkey(namespace, key, old, new)
367
367
368 def stream_out(self):
368 def stream_out(self):
369 raise error.Abort(_(b'cannot perform stream clone against local peer'))
369 raise error.Abort(_(b'cannot perform stream clone against local peer'))
370
370
371 def unbundle(self, bundle, heads, url):
371 def unbundle(self, bundle, heads, url):
372 """apply a bundle on a repo
372 """apply a bundle on a repo
373
373
374 This function handles the repo locking itself."""
374 This function handles the repo locking itself."""
375 try:
375 try:
376 try:
376 try:
377 bundle = exchange.readbundle(self.ui, bundle, None)
377 bundle = exchange.readbundle(self.ui, bundle, None)
378 ret = exchange.unbundle(self._repo, bundle, heads, b'push', url)
378 ret = exchange.unbundle(self._repo, bundle, heads, b'push', url)
379 if util.safehasattr(ret, b'getchunks'):
379 if util.safehasattr(ret, b'getchunks'):
380 # This is a bundle20 object, turn it into an unbundler.
380 # This is a bundle20 object, turn it into an unbundler.
381 # This little dance should be dropped eventually when the
381 # This little dance should be dropped eventually when the
382 # API is finally improved.
382 # API is finally improved.
383 stream = util.chunkbuffer(ret.getchunks())
383 stream = util.chunkbuffer(ret.getchunks())
384 ret = bundle2.getunbundler(self.ui, stream)
384 ret = bundle2.getunbundler(self.ui, stream)
385 return ret
385 return ret
386 except Exception as exc:
386 except Exception as exc:
387 # If the exception contains output salvaged from a bundle2
387 # If the exception contains output salvaged from a bundle2
388 # reply, we need to make sure it is printed before continuing
388 # reply, we need to make sure it is printed before continuing
389 # to fail. So we build a bundle2 with such output and consume
389 # to fail. So we build a bundle2 with such output and consume
390 # it directly.
390 # it directly.
391 #
391 #
392 # This is not very elegant but allows a "simple" solution for
392 # This is not very elegant but allows a "simple" solution for
393 # issue4594
393 # issue4594
394 output = getattr(exc, '_bundle2salvagedoutput', ())
394 output = getattr(exc, '_bundle2salvagedoutput', ())
395 if output:
395 if output:
396 bundler = bundle2.bundle20(self._repo.ui)
396 bundler = bundle2.bundle20(self._repo.ui)
397 for out in output:
397 for out in output:
398 bundler.addpart(out)
398 bundler.addpart(out)
399 stream = util.chunkbuffer(bundler.getchunks())
399 stream = util.chunkbuffer(bundler.getchunks())
400 b = bundle2.getunbundler(self.ui, stream)
400 b = bundle2.getunbundler(self.ui, stream)
401 bundle2.processbundle(self._repo, b)
401 bundle2.processbundle(self._repo, b)
402 raise
402 raise
403 except error.PushRaced as exc:
403 except error.PushRaced as exc:
404 raise error.ResponseError(
404 raise error.ResponseError(
405 _(b'push failed:'), stringutil.forcebytestr(exc)
405 _(b'push failed:'), stringutil.forcebytestr(exc)
406 )
406 )
407
407
408 # End of _basewirecommands interface.
408 # End of _basewirecommands interface.
409
409
410 # Begin of peer interface.
410 # Begin of peer interface.
411
411
412 def commandexecutor(self):
412 def commandexecutor(self):
413 return localcommandexecutor(self)
413 return localcommandexecutor(self)
414
414
415 # End of peer interface.
415 # End of peer interface.
416
416
417
417
418 @interfaceutil.implementer(repository.ipeerlegacycommands)
418 @interfaceutil.implementer(repository.ipeerlegacycommands)
419 class locallegacypeer(localpeer):
419 class locallegacypeer(localpeer):
420 """peer extension which implements legacy methods too; used for tests with
420 """peer extension which implements legacy methods too; used for tests with
421 restricted capabilities"""
421 restricted capabilities"""
422
422
423 def __init__(self, repo):
423 def __init__(self, repo):
424 super(locallegacypeer, self).__init__(repo, caps=legacycaps)
424 super(locallegacypeer, self).__init__(repo, caps=legacycaps)
425
425
426 # Begin of baselegacywirecommands interface.
426 # Begin of baselegacywirecommands interface.
427
427
428 def between(self, pairs):
428 def between(self, pairs):
429 return self._repo.between(pairs)
429 return self._repo.between(pairs)
430
430
431 def branches(self, nodes):
431 def branches(self, nodes):
432 return self._repo.branches(nodes)
432 return self._repo.branches(nodes)
433
433
434 def changegroup(self, nodes, source):
434 def changegroup(self, nodes, source):
435 outgoing = discovery.outgoing(
435 outgoing = discovery.outgoing(
436 self._repo, missingroots=nodes, ancestorsof=self._repo.heads()
436 self._repo, missingroots=nodes, ancestorsof=self._repo.heads()
437 )
437 )
438 return changegroup.makechangegroup(self._repo, outgoing, b'01', source)
438 return changegroup.makechangegroup(self._repo, outgoing, b'01', source)
439
439
440 def changegroupsubset(self, bases, heads, source):
440 def changegroupsubset(self, bases, heads, source):
441 outgoing = discovery.outgoing(
441 outgoing = discovery.outgoing(
442 self._repo, missingroots=bases, ancestorsof=heads
442 self._repo, missingroots=bases, ancestorsof=heads
443 )
443 )
444 return changegroup.makechangegroup(self._repo, outgoing, b'01', source)
444 return changegroup.makechangegroup(self._repo, outgoing, b'01', source)
445
445
446 # End of baselegacywirecommands interface.
446 # End of baselegacywirecommands interface.
447
447
448
448
449 # Functions receiving (ui, features) that extensions can register to impact
449 # Functions receiving (ui, features) that extensions can register to impact
450 # the ability to load repositories with custom requirements. Only
450 # the ability to load repositories with custom requirements. Only
451 # functions defined in loaded extensions are called.
451 # functions defined in loaded extensions are called.
452 #
452 #
453 # The function receives a set of requirement strings that the repository
453 # The function receives a set of requirement strings that the repository
454 # is capable of opening. Functions will typically add elements to the
454 # is capable of opening. Functions will typically add elements to the
455 # set to reflect that the extension knows how to handle that requirements.
455 # set to reflect that the extension knows how to handle that requirements.
456 featuresetupfuncs = set()
456 featuresetupfuncs = set()
457
457
458
458
459 def _getsharedvfs(hgvfs, requirements):
459 def _getsharedvfs(hgvfs, requirements):
460 """returns the vfs object pointing to root of shared source
460 """returns the vfs object pointing to root of shared source
461 repo for a shared repository
461 repo for a shared repository
462
462
463 hgvfs is vfs pointing at .hg/ of current repo (shared one)
463 hgvfs is vfs pointing at .hg/ of current repo (shared one)
464 requirements is a set of requirements of current repo (shared one)
464 requirements is a set of requirements of current repo (shared one)
465 """
465 """
466 # The ``shared`` or ``relshared`` requirements indicate the
466 # The ``shared`` or ``relshared`` requirements indicate the
467 # store lives in the path contained in the ``.hg/sharedpath`` file.
467 # store lives in the path contained in the ``.hg/sharedpath`` file.
468 # This is an absolute path for ``shared`` and relative to
468 # This is an absolute path for ``shared`` and relative to
469 # ``.hg/`` for ``relshared``.
469 # ``.hg/`` for ``relshared``.
470 sharedpath = hgvfs.read(b'sharedpath').rstrip(b'\n')
470 sharedpath = hgvfs.read(b'sharedpath').rstrip(b'\n')
471 if requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements:
471 if requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements:
472 sharedpath = hgvfs.join(sharedpath)
472 sharedpath = hgvfs.join(sharedpath)
473
473
474 sharedvfs = vfsmod.vfs(sharedpath, realpath=True)
474 sharedvfs = vfsmod.vfs(sharedpath, realpath=True)
475
475
476 if not sharedvfs.exists():
476 if not sharedvfs.exists():
477 raise error.RepoError(
477 raise error.RepoError(
478 _(b'.hg/sharedpath points to nonexistent directory %s')
478 _(b'.hg/sharedpath points to nonexistent directory %s')
479 % sharedvfs.base
479 % sharedvfs.base
480 )
480 )
481 return sharedvfs
481 return sharedvfs
482
482
483
483
484 def _readrequires(vfs, allowmissing):
484 def _readrequires(vfs, allowmissing):
485 """reads the require file present at root of this vfs
485 """reads the require file present at root of this vfs
486 and return a set of requirements
486 and return a set of requirements
487
487
488 If allowmissing is True, we suppress ENOENT if raised"""
488 If allowmissing is True, we suppress ENOENT if raised"""
489 # requires file contains a newline-delimited list of
489 # requires file contains a newline-delimited list of
490 # features/capabilities the opener (us) must have in order to use
490 # features/capabilities the opener (us) must have in order to use
491 # the repository. This file was introduced in Mercurial 0.9.2,
491 # the repository. This file was introduced in Mercurial 0.9.2,
492 # which means very old repositories may not have one. We assume
492 # which means very old repositories may not have one. We assume
493 # a missing file translates to no requirements.
493 # a missing file translates to no requirements.
494 try:
494 try:
495 requirements = set(vfs.read(b'requires').splitlines())
495 requirements = set(vfs.read(b'requires').splitlines())
496 except IOError as e:
496 except IOError as e:
497 if not (allowmissing and e.errno == errno.ENOENT):
497 if not (allowmissing and e.errno == errno.ENOENT):
498 raise
498 raise
499 requirements = set()
499 requirements = set()
500 return requirements
500 return requirements
501
501
502
502
503 def makelocalrepository(baseui, path, intents=None):
503 def makelocalrepository(baseui, path, intents=None):
504 """Create a local repository object.
504 """Create a local repository object.
505
505
506 Given arguments needed to construct a local repository, this function
506 Given arguments needed to construct a local repository, this function
507 performs various early repository loading functionality (such as
507 performs various early repository loading functionality (such as
508 reading the ``.hg/requires`` and ``.hg/hgrc`` files), validates that
508 reading the ``.hg/requires`` and ``.hg/hgrc`` files), validates that
509 the repository can be opened, derives a type suitable for representing
509 the repository can be opened, derives a type suitable for representing
510 that repository, and returns an instance of it.
510 that repository, and returns an instance of it.
511
511
512 The returned object conforms to the ``repository.completelocalrepository``
512 The returned object conforms to the ``repository.completelocalrepository``
513 interface.
513 interface.
514
514
515 The repository type is derived by calling a series of factory functions
515 The repository type is derived by calling a series of factory functions
516 for each aspect/interface of the final repository. These are defined by
516 for each aspect/interface of the final repository. These are defined by
517 ``REPO_INTERFACES``.
517 ``REPO_INTERFACES``.
518
518
519 Each factory function is called to produce a type implementing a specific
519 Each factory function is called to produce a type implementing a specific
520 interface. The cumulative list of returned types will be combined into a
520 interface. The cumulative list of returned types will be combined into a
521 new type and that type will be instantiated to represent the local
521 new type and that type will be instantiated to represent the local
522 repository.
522 repository.
523
523
524 The factory functions each receive various state that may be consulted
524 The factory functions each receive various state that may be consulted
525 as part of deriving a type.
525 as part of deriving a type.
526
526
527 Extensions should wrap these factory functions to customize repository type
527 Extensions should wrap these factory functions to customize repository type
528 creation. Note that an extension's wrapped function may be called even if
528 creation. Note that an extension's wrapped function may be called even if
529 that extension is not loaded for the repo being constructed. Extensions
529 that extension is not loaded for the repo being constructed. Extensions
530 should check if their ``__name__`` appears in the
530 should check if their ``__name__`` appears in the
531 ``extensionmodulenames`` set passed to the factory function and no-op if
531 ``extensionmodulenames`` set passed to the factory function and no-op if
532 not.
532 not.
533 """
533 """
534 ui = baseui.copy()
534 ui = baseui.copy()
535 # Prevent copying repo configuration.
535 # Prevent copying repo configuration.
536 ui.copy = baseui.copy
536 ui.copy = baseui.copy
537
537
538 # Working directory VFS rooted at repository root.
538 # Working directory VFS rooted at repository root.
539 wdirvfs = vfsmod.vfs(path, expandpath=True, realpath=True)
539 wdirvfs = vfsmod.vfs(path, expandpath=True, realpath=True)
540
540
541 # Main VFS for .hg/ directory.
541 # Main VFS for .hg/ directory.
542 hgpath = wdirvfs.join(b'.hg')
542 hgpath = wdirvfs.join(b'.hg')
543 hgvfs = vfsmod.vfs(hgpath, cacheaudited=True)
543 hgvfs = vfsmod.vfs(hgpath, cacheaudited=True)
544 # Whether this repository is shared one or not
544 # Whether this repository is shared one or not
545 shared = False
545 shared = False
546 # If this repository is shared, vfs pointing to shared repo
546 # If this repository is shared, vfs pointing to shared repo
547 sharedvfs = None
547 sharedvfs = None
548
548
549 # The .hg/ path should exist and should be a directory. All other
549 # The .hg/ path should exist and should be a directory. All other
550 # cases are errors.
550 # cases are errors.
551 if not hgvfs.isdir():
551 if not hgvfs.isdir():
552 try:
552 try:
553 hgvfs.stat()
553 hgvfs.stat()
554 except OSError as e:
554 except OSError as e:
555 if e.errno != errno.ENOENT:
555 if e.errno != errno.ENOENT:
556 raise
556 raise
557 except ValueError as e:
557 except ValueError as e:
558 # Can be raised on Python 3.8 when path is invalid.
558 # Can be raised on Python 3.8 when path is invalid.
559 raise error.Abort(
559 raise error.Abort(
560 _(b'invalid path %s: %s') % (path, stringutil.forcebytestr(e))
560 _(b'invalid path %s: %s') % (path, stringutil.forcebytestr(e))
561 )
561 )
562
562
563 raise error.RepoError(_(b'repository %s not found') % path)
563 raise error.RepoError(_(b'repository %s not found') % path)
564
564
565 requirements = _readrequires(hgvfs, True)
565 requirements = _readrequires(hgvfs, True)
566 shared = (
566 shared = (
567 requirementsmod.SHARED_REQUIREMENT in requirements
567 requirementsmod.SHARED_REQUIREMENT in requirements
568 or requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements
568 or requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements
569 )
569 )
570 storevfs = None
570 storevfs = None
571 if shared:
571 if shared:
572 # This is a shared repo
572 # This is a shared repo
573 sharedvfs = _getsharedvfs(hgvfs, requirements)
573 sharedvfs = _getsharedvfs(hgvfs, requirements)
574 storevfs = vfsmod.vfs(sharedvfs.join(b'store'))
574 storevfs = vfsmod.vfs(sharedvfs.join(b'store'))
575 else:
575 else:
576 storevfs = vfsmod.vfs(hgvfs.join(b'store'))
576 storevfs = vfsmod.vfs(hgvfs.join(b'store'))
577
577
578 # if .hg/requires contains the sharesafe requirement, it means
578 # if .hg/requires contains the sharesafe requirement, it means
579 # there exists a `.hg/store/requires` too and we should read it
579 # there exists a `.hg/store/requires` too and we should read it
580 # NOTE: presence of SHARESAFE_REQUIREMENT imply that store requirement
580 # NOTE: presence of SHARESAFE_REQUIREMENT imply that store requirement
581 # is present. We never write SHARESAFE_REQUIREMENT for a repo if store
581 # is present. We never write SHARESAFE_REQUIREMENT for a repo if store
582 # is not present, refer checkrequirementscompat() for that
582 # is not present, refer checkrequirementscompat() for that
583 #
583 #
584 # However, if SHARESAFE_REQUIREMENT is not present, it means that the
584 # However, if SHARESAFE_REQUIREMENT is not present, it means that the
585 # repository was shared the old way. We check the share source .hg/requires
585 # repository was shared the old way. We check the share source .hg/requires
586 # for SHARESAFE_REQUIREMENT to detect whether the current repository needs
586 # for SHARESAFE_REQUIREMENT to detect whether the current repository needs
587 # to be reshared
587 # to be reshared
588 hint = _(b"see `hg help config.format.use-share-safe` for more information")
588 hint = _(b"see `hg help config.format.use-share-safe` for more information")
589 if requirementsmod.SHARESAFE_REQUIREMENT in requirements:
589 if requirementsmod.SHARESAFE_REQUIREMENT in requirements:
590
590
591 if (
591 if (
592 shared
592 shared
593 and requirementsmod.SHARESAFE_REQUIREMENT
593 and requirementsmod.SHARESAFE_REQUIREMENT
594 not in _readrequires(sharedvfs, True)
594 not in _readrequires(sharedvfs, True)
595 ):
595 ):
596 mismatch_warn = ui.configbool(
596 mismatch_warn = ui.configbool(
597 b'share', b'safe-mismatch.source-not-safe.warn'
597 b'share', b'safe-mismatch.source-not-safe.warn'
598 )
598 )
599 mismatch_config = ui.config(
599 mismatch_config = ui.config(
600 b'share', b'safe-mismatch.source-not-safe'
600 b'share', b'safe-mismatch.source-not-safe'
601 )
601 )
602 if mismatch_config in (
602 if mismatch_config in (
603 b'downgrade-allow',
603 b'downgrade-allow',
604 b'allow',
604 b'allow',
605 b'downgrade-abort',
605 b'downgrade-abort',
606 ):
606 ):
607 # prevent cyclic import localrepo -> upgrade -> localrepo
607 # prevent cyclic import localrepo -> upgrade -> localrepo
608 from . import upgrade
608 from . import upgrade
609
609
610 upgrade.downgrade_share_to_non_safe(
610 upgrade.downgrade_share_to_non_safe(
611 ui,
611 ui,
612 hgvfs,
612 hgvfs,
613 sharedvfs,
613 sharedvfs,
614 requirements,
614 requirements,
615 mismatch_config,
615 mismatch_config,
616 mismatch_warn,
616 mismatch_warn,
617 )
617 )
618 elif mismatch_config == b'abort':
618 elif mismatch_config == b'abort':
619 raise error.Abort(
619 raise error.Abort(
620 _(b"share source does not support share-safe requirement"),
620 _(b"share source does not support share-safe requirement"),
621 hint=hint,
621 hint=hint,
622 )
622 )
623 else:
623 else:
624 raise error.Abort(
624 raise error.Abort(
625 _(
625 _(
626 b"share-safe mismatch with source.\nUnrecognized"
626 b"share-safe mismatch with source.\nUnrecognized"
627 b" value '%s' of `share.safe-mismatch.source-not-safe`"
627 b" value '%s' of `share.safe-mismatch.source-not-safe`"
628 b" set."
628 b" set."
629 )
629 )
630 % mismatch_config,
630 % mismatch_config,
631 hint=hint,
631 hint=hint,
632 )
632 )
633 else:
633 else:
634 requirements |= _readrequires(storevfs, False)
634 requirements |= _readrequires(storevfs, False)
635 elif shared:
635 elif shared:
636 sourcerequires = _readrequires(sharedvfs, False)
636 sourcerequires = _readrequires(sharedvfs, False)
637 if requirementsmod.SHARESAFE_REQUIREMENT in sourcerequires:
637 if requirementsmod.SHARESAFE_REQUIREMENT in sourcerequires:
638 mismatch_config = ui.config(b'share', b'safe-mismatch.source-safe')
638 mismatch_config = ui.config(b'share', b'safe-mismatch.source-safe')
639 mismatch_warn = ui.configbool(
639 mismatch_warn = ui.configbool(
640 b'share', b'safe-mismatch.source-safe.warn'
640 b'share', b'safe-mismatch.source-safe.warn'
641 )
641 )
642 if mismatch_config in (
642 if mismatch_config in (
643 b'upgrade-allow',
643 b'upgrade-allow',
644 b'allow',
644 b'allow',
645 b'upgrade-abort',
645 b'upgrade-abort',
646 ):
646 ):
647 # prevent cyclic import localrepo -> upgrade -> localrepo
647 # prevent cyclic import localrepo -> upgrade -> localrepo
648 from . import upgrade
648 from . import upgrade
649
649
650 upgrade.upgrade_share_to_safe(
650 upgrade.upgrade_share_to_safe(
651 ui,
651 ui,
652 hgvfs,
652 hgvfs,
653 storevfs,
653 storevfs,
654 requirements,
654 requirements,
655 mismatch_config,
655 mismatch_config,
656 mismatch_warn,
656 mismatch_warn,
657 )
657 )
658 elif mismatch_config == b'abort':
658 elif mismatch_config == b'abort':
659 raise error.Abort(
659 raise error.Abort(
660 _(
660 _(
661 b'version mismatch: source uses share-safe'
661 b'version mismatch: source uses share-safe'
662 b' functionality while the current share does not'
662 b' functionality while the current share does not'
663 ),
663 ),
664 hint=hint,
664 hint=hint,
665 )
665 )
666 else:
666 else:
667 raise error.Abort(
667 raise error.Abort(
668 _(
668 _(
669 b"share-safe mismatch with source.\nUnrecognized"
669 b"share-safe mismatch with source.\nUnrecognized"
670 b" value '%s' of `share.safe-mismatch.source-safe` set."
670 b" value '%s' of `share.safe-mismatch.source-safe` set."
671 )
671 )
672 % mismatch_config,
672 % mismatch_config,
673 hint=hint,
673 hint=hint,
674 )
674 )
675
675
676 # The .hg/hgrc file may load extensions or contain config options
676 # The .hg/hgrc file may load extensions or contain config options
677 # that influence repository construction. Attempt to load it and
677 # that influence repository construction. Attempt to load it and
678 # process any new extensions that it may have pulled in.
678 # process any new extensions that it may have pulled in.
679 if loadhgrc(ui, wdirvfs, hgvfs, requirements, sharedvfs):
679 if loadhgrc(ui, wdirvfs, hgvfs, requirements, sharedvfs):
680 afterhgrcload(ui, wdirvfs, hgvfs, requirements)
680 afterhgrcload(ui, wdirvfs, hgvfs, requirements)
681 extensions.loadall(ui)
681 extensions.loadall(ui)
682 extensions.populateui(ui)
682 extensions.populateui(ui)
683
683
684 # Set of module names of extensions loaded for this repository.
684 # Set of module names of extensions loaded for this repository.
685 extensionmodulenames = {m.__name__ for n, m in extensions.extensions(ui)}
685 extensionmodulenames = {m.__name__ for n, m in extensions.extensions(ui)}
686
686
687 supportedrequirements = gathersupportedrequirements(ui)
687 supportedrequirements = gathersupportedrequirements(ui)
688
688
689 # We first validate the requirements are known.
689 # We first validate the requirements are known.
690 ensurerequirementsrecognized(requirements, supportedrequirements)
690 ensurerequirementsrecognized(requirements, supportedrequirements)
691
691
692 # Then we validate that the known set is reasonable to use together.
692 # Then we validate that the known set is reasonable to use together.
693 ensurerequirementscompatible(ui, requirements)
693 ensurerequirementscompatible(ui, requirements)
694
694
695 # TODO there are unhandled edge cases related to opening repositories with
695 # TODO there are unhandled edge cases related to opening repositories with
696 # shared storage. If storage is shared, we should also test for requirements
696 # shared storage. If storage is shared, we should also test for requirements
697 # compatibility in the pointed-to repo. This entails loading the .hg/hgrc in
697 # compatibility in the pointed-to repo. This entails loading the .hg/hgrc in
698 # that repo, as that repo may load extensions needed to open it. This is a
698 # that repo, as that repo may load extensions needed to open it. This is a
699 # bit complicated because we don't want the other hgrc to overwrite settings
699 # bit complicated because we don't want the other hgrc to overwrite settings
700 # in this hgrc.
700 # in this hgrc.
701 #
701 #
702 # This bug is somewhat mitigated by the fact that we copy the .hg/requires
702 # This bug is somewhat mitigated by the fact that we copy the .hg/requires
703 # file when sharing repos. But if a requirement is added after the share is
703 # file when sharing repos. But if a requirement is added after the share is
704 # performed, thereby introducing a new requirement for the opener, we may
704 # performed, thereby introducing a new requirement for the opener, we may
705 # will not see that and could encounter a run-time error interacting with
705 # will not see that and could encounter a run-time error interacting with
706 # that shared store since it has an unknown-to-us requirement.
706 # that shared store since it has an unknown-to-us requirement.
707
707
708 # At this point, we know we should be capable of opening the repository.
708 # At this point, we know we should be capable of opening the repository.
709 # Now get on with doing that.
709 # Now get on with doing that.
710
710
711 features = set()
711 features = set()
712
712
713 # The "store" part of the repository holds versioned data. How it is
713 # The "store" part of the repository holds versioned data. How it is
714 # accessed is determined by various requirements. If `shared` or
714 # accessed is determined by various requirements. If `shared` or
715 # `relshared` requirements are present, this indicates current repository
715 # `relshared` requirements are present, this indicates current repository
716 # is a share and store exists in path mentioned in `.hg/sharedpath`
716 # is a share and store exists in path mentioned in `.hg/sharedpath`
717 if shared:
717 if shared:
718 storebasepath = sharedvfs.base
718 storebasepath = sharedvfs.base
719 cachepath = sharedvfs.join(b'cache')
719 cachepath = sharedvfs.join(b'cache')
720 features.add(repository.REPO_FEATURE_SHARED_STORAGE)
720 features.add(repository.REPO_FEATURE_SHARED_STORAGE)
721 else:
721 else:
722 storebasepath = hgvfs.base
722 storebasepath = hgvfs.base
723 cachepath = hgvfs.join(b'cache')
723 cachepath = hgvfs.join(b'cache')
724 wcachepath = hgvfs.join(b'wcache')
724 wcachepath = hgvfs.join(b'wcache')
725
725
726 # The store has changed over time and the exact layout is dictated by
726 # The store has changed over time and the exact layout is dictated by
727 # requirements. The store interface abstracts differences across all
727 # requirements. The store interface abstracts differences across all
728 # of them.
728 # of them.
729 store = makestore(
729 store = makestore(
730 requirements,
730 requirements,
731 storebasepath,
731 storebasepath,
732 lambda base: vfsmod.vfs(base, cacheaudited=True),
732 lambda base: vfsmod.vfs(base, cacheaudited=True),
733 )
733 )
734 hgvfs.createmode = store.createmode
734 hgvfs.createmode = store.createmode
735
735
736 storevfs = store.vfs
736 storevfs = store.vfs
737 storevfs.options = resolvestorevfsoptions(ui, requirements, features)
737 storevfs.options = resolvestorevfsoptions(ui, requirements, features)
738
738
739 # The cache vfs is used to manage cache files.
739 # The cache vfs is used to manage cache files.
740 cachevfs = vfsmod.vfs(cachepath, cacheaudited=True)
740 cachevfs = vfsmod.vfs(cachepath, cacheaudited=True)
741 cachevfs.createmode = store.createmode
741 cachevfs.createmode = store.createmode
742 # The cache vfs is used to manage cache files related to the working copy
742 # The cache vfs is used to manage cache files related to the working copy
743 wcachevfs = vfsmod.vfs(wcachepath, cacheaudited=True)
743 wcachevfs = vfsmod.vfs(wcachepath, cacheaudited=True)
744 wcachevfs.createmode = store.createmode
744 wcachevfs.createmode = store.createmode
745
745
746 # Now resolve the type for the repository object. We do this by repeatedly
746 # Now resolve the type for the repository object. We do this by repeatedly
747 # calling a factory function to produces types for specific aspects of the
747 # calling a factory function to produces types for specific aspects of the
748 # repo's operation. The aggregate returned types are used as base classes
748 # repo's operation. The aggregate returned types are used as base classes
749 # for a dynamically-derived type, which will represent our new repository.
749 # for a dynamically-derived type, which will represent our new repository.
750
750
751 bases = []
751 bases = []
752 extrastate = {}
752 extrastate = {}
753
753
754 for iface, fn in REPO_INTERFACES:
754 for iface, fn in REPO_INTERFACES:
755 # We pass all potentially useful state to give extensions tons of
755 # We pass all potentially useful state to give extensions tons of
756 # flexibility.
756 # flexibility.
757 typ = fn()(
757 typ = fn()(
758 ui=ui,
758 ui=ui,
759 intents=intents,
759 intents=intents,
760 requirements=requirements,
760 requirements=requirements,
761 features=features,
761 features=features,
762 wdirvfs=wdirvfs,
762 wdirvfs=wdirvfs,
763 hgvfs=hgvfs,
763 hgvfs=hgvfs,
764 store=store,
764 store=store,
765 storevfs=storevfs,
765 storevfs=storevfs,
766 storeoptions=storevfs.options,
766 storeoptions=storevfs.options,
767 cachevfs=cachevfs,
767 cachevfs=cachevfs,
768 wcachevfs=wcachevfs,
768 wcachevfs=wcachevfs,
769 extensionmodulenames=extensionmodulenames,
769 extensionmodulenames=extensionmodulenames,
770 extrastate=extrastate,
770 extrastate=extrastate,
771 baseclasses=bases,
771 baseclasses=bases,
772 )
772 )
773
773
774 if not isinstance(typ, type):
774 if not isinstance(typ, type):
775 raise error.ProgrammingError(
775 raise error.ProgrammingError(
776 b'unable to construct type for %s' % iface
776 b'unable to construct type for %s' % iface
777 )
777 )
778
778
779 bases.append(typ)
779 bases.append(typ)
780
780
781 # type() allows you to use characters in type names that wouldn't be
781 # type() allows you to use characters in type names that wouldn't be
782 # recognized as Python symbols in source code. We abuse that to add
782 # recognized as Python symbols in source code. We abuse that to add
783 # rich information about our constructed repo.
783 # rich information about our constructed repo.
784 name = pycompat.sysstr(
784 name = pycompat.sysstr(
785 b'derivedrepo:%s<%s>' % (wdirvfs.base, b','.join(sorted(requirements)))
785 b'derivedrepo:%s<%s>' % (wdirvfs.base, b','.join(sorted(requirements)))
786 )
786 )
787
787
788 cls = type(name, tuple(bases), {})
788 cls = type(name, tuple(bases), {})
789
789
790 return cls(
790 return cls(
791 baseui=baseui,
791 baseui=baseui,
792 ui=ui,
792 ui=ui,
793 origroot=path,
793 origroot=path,
794 wdirvfs=wdirvfs,
794 wdirvfs=wdirvfs,
795 hgvfs=hgvfs,
795 hgvfs=hgvfs,
796 requirements=requirements,
796 requirements=requirements,
797 supportedrequirements=supportedrequirements,
797 supportedrequirements=supportedrequirements,
798 sharedpath=storebasepath,
798 sharedpath=storebasepath,
799 store=store,
799 store=store,
800 cachevfs=cachevfs,
800 cachevfs=cachevfs,
801 wcachevfs=wcachevfs,
801 wcachevfs=wcachevfs,
802 features=features,
802 features=features,
803 intents=intents,
803 intents=intents,
804 )
804 )
805
805
806
806
807 def loadhgrc(ui, wdirvfs, hgvfs, requirements, sharedvfs=None):
807 def loadhgrc(ui, wdirvfs, hgvfs, requirements, sharedvfs=None):
808 """Load hgrc files/content into a ui instance.
808 """Load hgrc files/content into a ui instance.
809
809
810 This is called during repository opening to load any additional
810 This is called during repository opening to load any additional
811 config files or settings relevant to the current repository.
811 config files or settings relevant to the current repository.
812
812
813 Returns a bool indicating whether any additional configs were loaded.
813 Returns a bool indicating whether any additional configs were loaded.
814
814
815 Extensions should monkeypatch this function to modify how per-repo
815 Extensions should monkeypatch this function to modify how per-repo
816 configs are loaded. For example, an extension may wish to pull in
816 configs are loaded. For example, an extension may wish to pull in
817 configs from alternate files or sources.
817 configs from alternate files or sources.
818
818
819 sharedvfs is vfs object pointing to source repo if the current one is a
819 sharedvfs is vfs object pointing to source repo if the current one is a
820 shared one
820 shared one
821 """
821 """
822 if not rcutil.use_repo_hgrc():
822 if not rcutil.use_repo_hgrc():
823 return False
823 return False
824
824
825 ret = False
825 ret = False
826 # first load config from shared source if we has to
826 # first load config from shared source if we has to
827 if requirementsmod.SHARESAFE_REQUIREMENT in requirements and sharedvfs:
827 if requirementsmod.SHARESAFE_REQUIREMENT in requirements and sharedvfs:
828 try:
828 try:
829 ui.readconfig(sharedvfs.join(b'hgrc'), root=sharedvfs.base)
829 ui.readconfig(sharedvfs.join(b'hgrc'), root=sharedvfs.base)
830 ret = True
830 ret = True
831 except IOError:
831 except IOError:
832 pass
832 pass
833
833
834 try:
834 try:
835 ui.readconfig(hgvfs.join(b'hgrc'), root=wdirvfs.base)
835 ui.readconfig(hgvfs.join(b'hgrc'), root=wdirvfs.base)
836 ret = True
836 ret = True
837 except IOError:
837 except IOError:
838 pass
838 pass
839
839
840 try:
840 try:
841 ui.readconfig(hgvfs.join(b'hgrc-not-shared'), root=wdirvfs.base)
841 ui.readconfig(hgvfs.join(b'hgrc-not-shared'), root=wdirvfs.base)
842 ret = True
842 ret = True
843 except IOError:
843 except IOError:
844 pass
844 pass
845
845
846 return ret
846 return ret
847
847
848
848
849 def afterhgrcload(ui, wdirvfs, hgvfs, requirements):
849 def afterhgrcload(ui, wdirvfs, hgvfs, requirements):
850 """Perform additional actions after .hg/hgrc is loaded.
850 """Perform additional actions after .hg/hgrc is loaded.
851
851
852 This function is called during repository loading immediately after
852 This function is called during repository loading immediately after
853 the .hg/hgrc file is loaded and before per-repo extensions are loaded.
853 the .hg/hgrc file is loaded and before per-repo extensions are loaded.
854
854
855 The function can be used to validate configs, automatically add
855 The function can be used to validate configs, automatically add
856 options (including extensions) based on requirements, etc.
856 options (including extensions) based on requirements, etc.
857 """
857 """
858
858
859 # Map of requirements to list of extensions to load automatically when
859 # Map of requirements to list of extensions to load automatically when
860 # requirement is present.
860 # requirement is present.
861 autoextensions = {
861 autoextensions = {
862 b'git': [b'git'],
862 b'git': [b'git'],
863 b'largefiles': [b'largefiles'],
863 b'largefiles': [b'largefiles'],
864 b'lfs': [b'lfs'],
864 b'lfs': [b'lfs'],
865 }
865 }
866
866
867 for requirement, names in sorted(autoextensions.items()):
867 for requirement, names in sorted(autoextensions.items()):
868 if requirement not in requirements:
868 if requirement not in requirements:
869 continue
869 continue
870
870
871 for name in names:
871 for name in names:
872 if not ui.hasconfig(b'extensions', name):
872 if not ui.hasconfig(b'extensions', name):
873 ui.setconfig(b'extensions', name, b'', source=b'autoload')
873 ui.setconfig(b'extensions', name, b'', source=b'autoload')
874
874
875
875
876 def gathersupportedrequirements(ui):
876 def gathersupportedrequirements(ui):
877 """Determine the complete set of recognized requirements."""
877 """Determine the complete set of recognized requirements."""
878 # Start with all requirements supported by this file.
878 # Start with all requirements supported by this file.
879 supported = set(localrepository._basesupported)
879 supported = set(localrepository._basesupported)
880
880
881 # Execute ``featuresetupfuncs`` entries if they belong to an extension
881 # Execute ``featuresetupfuncs`` entries if they belong to an extension
882 # relevant to this ui instance.
882 # relevant to this ui instance.
883 modules = {m.__name__ for n, m in extensions.extensions(ui)}
883 modules = {m.__name__ for n, m in extensions.extensions(ui)}
884
884
885 for fn in featuresetupfuncs:
885 for fn in featuresetupfuncs:
886 if fn.__module__ in modules:
886 if fn.__module__ in modules:
887 fn(ui, supported)
887 fn(ui, supported)
888
888
889 # Add derived requirements from registered compression engines.
889 # Add derived requirements from registered compression engines.
890 for name in util.compengines:
890 for name in util.compengines:
891 engine = util.compengines[name]
891 engine = util.compengines[name]
892 if engine.available() and engine.revlogheader():
892 if engine.available() and engine.revlogheader():
893 supported.add(b'exp-compression-%s' % name)
893 supported.add(b'exp-compression-%s' % name)
894 if engine.name() == b'zstd':
894 if engine.name() == b'zstd':
895 supported.add(b'revlog-compression-zstd')
895 supported.add(b'revlog-compression-zstd')
896
896
897 return supported
897 return supported
898
898
899
899
900 def ensurerequirementsrecognized(requirements, supported):
900 def ensurerequirementsrecognized(requirements, supported):
901 """Validate that a set of local requirements is recognized.
901 """Validate that a set of local requirements is recognized.
902
902
903 Receives a set of requirements. Raises an ``error.RepoError`` if there
903 Receives a set of requirements. Raises an ``error.RepoError`` if there
904 exists any requirement in that set that currently loaded code doesn't
904 exists any requirement in that set that currently loaded code doesn't
905 recognize.
905 recognize.
906
906
907 Returns a set of supported requirements.
907 Returns a set of supported requirements.
908 """
908 """
909 missing = set()
909 missing = set()
910
910
911 for requirement in requirements:
911 for requirement in requirements:
912 if requirement in supported:
912 if requirement in supported:
913 continue
913 continue
914
914
915 if not requirement or not requirement[0:1].isalnum():
915 if not requirement or not requirement[0:1].isalnum():
916 raise error.RequirementError(_(b'.hg/requires file is corrupt'))
916 raise error.RequirementError(_(b'.hg/requires file is corrupt'))
917
917
918 missing.add(requirement)
918 missing.add(requirement)
919
919
920 if missing:
920 if missing:
921 raise error.RequirementError(
921 raise error.RequirementError(
922 _(b'repository requires features unknown to this Mercurial: %s')
922 _(b'repository requires features unknown to this Mercurial: %s')
923 % b' '.join(sorted(missing)),
923 % b' '.join(sorted(missing)),
924 hint=_(
924 hint=_(
925 b'see https://mercurial-scm.org/wiki/MissingRequirement '
925 b'see https://mercurial-scm.org/wiki/MissingRequirement '
926 b'for more information'
926 b'for more information'
927 ),
927 ),
928 )
928 )
929
929
930
930
931 def ensurerequirementscompatible(ui, requirements):
931 def ensurerequirementscompatible(ui, requirements):
932 """Validates that a set of recognized requirements is mutually compatible.
932 """Validates that a set of recognized requirements is mutually compatible.
933
933
934 Some requirements may not be compatible with others or require
934 Some requirements may not be compatible with others or require
935 config options that aren't enabled. This function is called during
935 config options that aren't enabled. This function is called during
936 repository opening to ensure that the set of requirements needed
936 repository opening to ensure that the set of requirements needed
937 to open a repository is sane and compatible with config options.
937 to open a repository is sane and compatible with config options.
938
938
939 Extensions can monkeypatch this function to perform additional
939 Extensions can monkeypatch this function to perform additional
940 checking.
940 checking.
941
941
942 ``error.RepoError`` should be raised on failure.
942 ``error.RepoError`` should be raised on failure.
943 """
943 """
944 if (
944 if (
945 requirementsmod.SPARSE_REQUIREMENT in requirements
945 requirementsmod.SPARSE_REQUIREMENT in requirements
946 and not sparse.enabled
946 and not sparse.enabled
947 ):
947 ):
948 raise error.RepoError(
948 raise error.RepoError(
949 _(
949 _(
950 b'repository is using sparse feature but '
950 b'repository is using sparse feature but '
951 b'sparse is not enabled; enable the '
951 b'sparse is not enabled; enable the '
952 b'"sparse" extensions to access'
952 b'"sparse" extensions to access'
953 )
953 )
954 )
954 )
955
955
956
956
957 def makestore(requirements, path, vfstype):
957 def makestore(requirements, path, vfstype):
958 """Construct a storage object for a repository."""
958 """Construct a storage object for a repository."""
959 if requirementsmod.STORE_REQUIREMENT in requirements:
959 if requirementsmod.STORE_REQUIREMENT in requirements:
960 if requirementsmod.FNCACHE_REQUIREMENT in requirements:
960 if requirementsmod.FNCACHE_REQUIREMENT in requirements:
961 dotencode = requirementsmod.DOTENCODE_REQUIREMENT in requirements
961 dotencode = requirementsmod.DOTENCODE_REQUIREMENT in requirements
962 return storemod.fncachestore(path, vfstype, dotencode)
962 return storemod.fncachestore(path, vfstype, dotencode)
963
963
964 return storemod.encodedstore(path, vfstype)
964 return storemod.encodedstore(path, vfstype)
965
965
966 return storemod.basicstore(path, vfstype)
966 return storemod.basicstore(path, vfstype)
967
967
968
968
969 def resolvestorevfsoptions(ui, requirements, features):
969 def resolvestorevfsoptions(ui, requirements, features):
970 """Resolve the options to pass to the store vfs opener.
970 """Resolve the options to pass to the store vfs opener.
971
971
972 The returned dict is used to influence behavior of the storage layer.
972 The returned dict is used to influence behavior of the storage layer.
973 """
973 """
974 options = {}
974 options = {}
975
975
976 if requirementsmod.TREEMANIFEST_REQUIREMENT in requirements:
976 if requirementsmod.TREEMANIFEST_REQUIREMENT in requirements:
977 options[b'treemanifest'] = True
977 options[b'treemanifest'] = True
978
978
979 # experimental config: format.manifestcachesize
979 # experimental config: format.manifestcachesize
980 manifestcachesize = ui.configint(b'format', b'manifestcachesize')
980 manifestcachesize = ui.configint(b'format', b'manifestcachesize')
981 if manifestcachesize is not None:
981 if manifestcachesize is not None:
982 options[b'manifestcachesize'] = manifestcachesize
982 options[b'manifestcachesize'] = manifestcachesize
983
983
984 # In the absence of another requirement superseding a revlog-related
984 # In the absence of another requirement superseding a revlog-related
985 # requirement, we have to assume the repo is using revlog version 0.
985 # requirement, we have to assume the repo is using revlog version 0.
986 # This revlog format is super old and we don't bother trying to parse
986 # This revlog format is super old and we don't bother trying to parse
987 # opener options for it because those options wouldn't do anything
987 # opener options for it because those options wouldn't do anything
988 # meaningful on such old repos.
988 # meaningful on such old repos.
989 if (
989 if (
990 requirementsmod.REVLOGV1_REQUIREMENT in requirements
990 requirementsmod.REVLOGV1_REQUIREMENT in requirements
991 or requirementsmod.REVLOGV2_REQUIREMENT in requirements
991 or requirementsmod.REVLOGV2_REQUIREMENT in requirements
992 ):
992 ):
993 options.update(resolverevlogstorevfsoptions(ui, requirements, features))
993 options.update(resolverevlogstorevfsoptions(ui, requirements, features))
994 else: # explicitly mark repo as using revlogv0
994 else: # explicitly mark repo as using revlogv0
995 options[b'revlogv0'] = True
995 options[b'revlogv0'] = True
996
996
997 if requirementsmod.COPIESSDC_REQUIREMENT in requirements:
997 if requirementsmod.COPIESSDC_REQUIREMENT in requirements:
998 options[b'copies-storage'] = b'changeset-sidedata'
998 options[b'copies-storage'] = b'changeset-sidedata'
999 else:
999 else:
1000 writecopiesto = ui.config(b'experimental', b'copies.write-to')
1000 writecopiesto = ui.config(b'experimental', b'copies.write-to')
1001 copiesextramode = (b'changeset-only', b'compatibility')
1001 copiesextramode = (b'changeset-only', b'compatibility')
1002 if writecopiesto in copiesextramode:
1002 if writecopiesto in copiesextramode:
1003 options[b'copies-storage'] = b'extra'
1003 options[b'copies-storage'] = b'extra'
1004
1004
1005 return options
1005 return options
1006
1006
1007
1007
1008 def resolverevlogstorevfsoptions(ui, requirements, features):
1008 def resolverevlogstorevfsoptions(ui, requirements, features):
1009 """Resolve opener options specific to revlogs."""
1009 """Resolve opener options specific to revlogs."""
1010
1010
1011 options = {}
1011 options = {}
1012 options[b'flagprocessors'] = {}
1012 options[b'flagprocessors'] = {}
1013
1013
1014 if requirementsmod.REVLOGV1_REQUIREMENT in requirements:
1014 if requirementsmod.REVLOGV1_REQUIREMENT in requirements:
1015 options[b'revlogv1'] = True
1015 options[b'revlogv1'] = True
1016 if requirementsmod.REVLOGV2_REQUIREMENT in requirements:
1016 if requirementsmod.REVLOGV2_REQUIREMENT in requirements:
1017 options[b'revlogv2'] = True
1017 options[b'revlogv2'] = True
1018
1018
1019 if requirementsmod.GENERALDELTA_REQUIREMENT in requirements:
1019 if requirementsmod.GENERALDELTA_REQUIREMENT in requirements:
1020 options[b'generaldelta'] = True
1020 options[b'generaldelta'] = True
1021
1021
1022 # experimental config: format.chunkcachesize
1022 # experimental config: format.chunkcachesize
1023 chunkcachesize = ui.configint(b'format', b'chunkcachesize')
1023 chunkcachesize = ui.configint(b'format', b'chunkcachesize')
1024 if chunkcachesize is not None:
1024 if chunkcachesize is not None:
1025 options[b'chunkcachesize'] = chunkcachesize
1025 options[b'chunkcachesize'] = chunkcachesize
1026
1026
1027 deltabothparents = ui.configbool(
1027 deltabothparents = ui.configbool(
1028 b'storage', b'revlog.optimize-delta-parent-choice'
1028 b'storage', b'revlog.optimize-delta-parent-choice'
1029 )
1029 )
1030 options[b'deltabothparents'] = deltabothparents
1030 options[b'deltabothparents'] = deltabothparents
1031
1031
1032 lazydelta = ui.configbool(b'storage', b'revlog.reuse-external-delta')
1032 lazydelta = ui.configbool(b'storage', b'revlog.reuse-external-delta')
1033 lazydeltabase = False
1033 lazydeltabase = False
1034 if lazydelta:
1034 if lazydelta:
1035 lazydeltabase = ui.configbool(
1035 lazydeltabase = ui.configbool(
1036 b'storage', b'revlog.reuse-external-delta-parent'
1036 b'storage', b'revlog.reuse-external-delta-parent'
1037 )
1037 )
1038 if lazydeltabase is None:
1038 if lazydeltabase is None:
1039 lazydeltabase = not scmutil.gddeltaconfig(ui)
1039 lazydeltabase = not scmutil.gddeltaconfig(ui)
1040 options[b'lazydelta'] = lazydelta
1040 options[b'lazydelta'] = lazydelta
1041 options[b'lazydeltabase'] = lazydeltabase
1041 options[b'lazydeltabase'] = lazydeltabase
1042
1042
1043 chainspan = ui.configbytes(b'experimental', b'maxdeltachainspan')
1043 chainspan = ui.configbytes(b'experimental', b'maxdeltachainspan')
1044 if 0 <= chainspan:
1044 if 0 <= chainspan:
1045 options[b'maxdeltachainspan'] = chainspan
1045 options[b'maxdeltachainspan'] = chainspan
1046
1046
1047 mmapindexthreshold = ui.configbytes(b'experimental', b'mmapindexthreshold')
1047 mmapindexthreshold = ui.configbytes(b'experimental', b'mmapindexthreshold')
1048 if mmapindexthreshold is not None:
1048 if mmapindexthreshold is not None:
1049 options[b'mmapindexthreshold'] = mmapindexthreshold
1049 options[b'mmapindexthreshold'] = mmapindexthreshold
1050
1050
1051 withsparseread = ui.configbool(b'experimental', b'sparse-read')
1051 withsparseread = ui.configbool(b'experimental', b'sparse-read')
1052 srdensitythres = float(
1052 srdensitythres = float(
1053 ui.config(b'experimental', b'sparse-read.density-threshold')
1053 ui.config(b'experimental', b'sparse-read.density-threshold')
1054 )
1054 )
1055 srmingapsize = ui.configbytes(b'experimental', b'sparse-read.min-gap-size')
1055 srmingapsize = ui.configbytes(b'experimental', b'sparse-read.min-gap-size')
1056 options[b'with-sparse-read'] = withsparseread
1056 options[b'with-sparse-read'] = withsparseread
1057 options[b'sparse-read-density-threshold'] = srdensitythres
1057 options[b'sparse-read-density-threshold'] = srdensitythres
1058 options[b'sparse-read-min-gap-size'] = srmingapsize
1058 options[b'sparse-read-min-gap-size'] = srmingapsize
1059
1059
1060 sparserevlog = requirementsmod.SPARSEREVLOG_REQUIREMENT in requirements
1060 sparserevlog = requirementsmod.SPARSEREVLOG_REQUIREMENT in requirements
1061 options[b'sparse-revlog'] = sparserevlog
1061 options[b'sparse-revlog'] = sparserevlog
1062 if sparserevlog:
1062 if sparserevlog:
1063 options[b'generaldelta'] = True
1063 options[b'generaldelta'] = True
1064
1064
1065 sidedata = requirementsmod.SIDEDATA_REQUIREMENT in requirements
1065 sidedata = requirementsmod.SIDEDATA_REQUIREMENT in requirements
1066 options[b'side-data'] = sidedata
1066 options[b'side-data'] = sidedata
1067
1067
1068 maxchainlen = None
1068 maxchainlen = None
1069 if sparserevlog:
1069 if sparserevlog:
1070 maxchainlen = revlogconst.SPARSE_REVLOG_MAX_CHAIN_LENGTH
1070 maxchainlen = revlogconst.SPARSE_REVLOG_MAX_CHAIN_LENGTH
1071 # experimental config: format.maxchainlen
1071 # experimental config: format.maxchainlen
1072 maxchainlen = ui.configint(b'format', b'maxchainlen', maxchainlen)
1072 maxchainlen = ui.configint(b'format', b'maxchainlen', maxchainlen)
1073 if maxchainlen is not None:
1073 if maxchainlen is not None:
1074 options[b'maxchainlen'] = maxchainlen
1074 options[b'maxchainlen'] = maxchainlen
1075
1075
1076 for r in requirements:
1076 for r in requirements:
1077 # we allow multiple compression engine requirement to co-exist because
1077 # we allow multiple compression engine requirement to co-exist because
1078 # strickly speaking, revlog seems to support mixed compression style.
1078 # strickly speaking, revlog seems to support mixed compression style.
1079 #
1079 #
1080 # The compression used for new entries will be "the last one"
1080 # The compression used for new entries will be "the last one"
1081 prefix = r.startswith
1081 prefix = r.startswith
1082 if prefix(b'revlog-compression-') or prefix(b'exp-compression-'):
1082 if prefix(b'revlog-compression-') or prefix(b'exp-compression-'):
1083 options[b'compengine'] = r.split(b'-', 2)[2]
1083 options[b'compengine'] = r.split(b'-', 2)[2]
1084
1084
1085 options[b'zlib.level'] = ui.configint(b'storage', b'revlog.zlib.level')
1085 options[b'zlib.level'] = ui.configint(b'storage', b'revlog.zlib.level')
1086 if options[b'zlib.level'] is not None:
1086 if options[b'zlib.level'] is not None:
1087 if not (0 <= options[b'zlib.level'] <= 9):
1087 if not (0 <= options[b'zlib.level'] <= 9):
1088 msg = _(b'invalid value for `storage.revlog.zlib.level` config: %d')
1088 msg = _(b'invalid value for `storage.revlog.zlib.level` config: %d')
1089 raise error.Abort(msg % options[b'zlib.level'])
1089 raise error.Abort(msg % options[b'zlib.level'])
1090 options[b'zstd.level'] = ui.configint(b'storage', b'revlog.zstd.level')
1090 options[b'zstd.level'] = ui.configint(b'storage', b'revlog.zstd.level')
1091 if options[b'zstd.level'] is not None:
1091 if options[b'zstd.level'] is not None:
1092 if not (0 <= options[b'zstd.level'] <= 22):
1092 if not (0 <= options[b'zstd.level'] <= 22):
1093 msg = _(b'invalid value for `storage.revlog.zstd.level` config: %d')
1093 msg = _(b'invalid value for `storage.revlog.zstd.level` config: %d')
1094 raise error.Abort(msg % options[b'zstd.level'])
1094 raise error.Abort(msg % options[b'zstd.level'])
1095
1095
1096 if requirementsmod.NARROW_REQUIREMENT in requirements:
1096 if requirementsmod.NARROW_REQUIREMENT in requirements:
1097 options[b'enableellipsis'] = True
1097 options[b'enableellipsis'] = True
1098
1098
1099 if ui.configbool(b'experimental', b'rust.index'):
1099 if ui.configbool(b'experimental', b'rust.index'):
1100 options[b'rust.index'] = True
1100 options[b'rust.index'] = True
1101 if requirementsmod.NODEMAP_REQUIREMENT in requirements:
1101 if requirementsmod.NODEMAP_REQUIREMENT in requirements:
1102 slow_path = ui.config(
1102 slow_path = ui.config(
1103 b'storage', b'revlog.persistent-nodemap.slow-path'
1103 b'storage', b'revlog.persistent-nodemap.slow-path'
1104 )
1104 )
1105 if slow_path not in (b'allow', b'warn', b'abort'):
1105 if slow_path not in (b'allow', b'warn', b'abort'):
1106 default = ui.config_default(
1106 default = ui.config_default(
1107 b'storage', b'revlog.persistent-nodemap.slow-path'
1107 b'storage', b'revlog.persistent-nodemap.slow-path'
1108 )
1108 )
1109 msg = _(
1109 msg = _(
1110 b'unknown value for config '
1110 b'unknown value for config '
1111 b'"storage.revlog.persistent-nodemap.slow-path": "%s"\n'
1111 b'"storage.revlog.persistent-nodemap.slow-path": "%s"\n'
1112 )
1112 )
1113 ui.warn(msg % slow_path)
1113 ui.warn(msg % slow_path)
1114 if not ui.quiet:
1114 if not ui.quiet:
1115 ui.warn(_(b'falling back to default value: %s\n') % default)
1115 ui.warn(_(b'falling back to default value: %s\n') % default)
1116 slow_path = default
1116 slow_path = default
1117
1117
1118 msg = _(
1118 msg = _(
1119 b"accessing `persistent-nodemap` repository without associated "
1119 b"accessing `persistent-nodemap` repository without associated "
1120 b"fast implementation."
1120 b"fast implementation."
1121 )
1121 )
1122 hint = _(
1122 hint = _(
1123 b"check `hg help config.format.use-persistent-nodemap` "
1123 b"check `hg help config.format.use-persistent-nodemap` "
1124 b"for details"
1124 b"for details"
1125 )
1125 )
1126 if not revlog.HAS_FAST_PERSISTENT_NODEMAP:
1126 if not revlog.HAS_FAST_PERSISTENT_NODEMAP:
1127 if slow_path == b'warn':
1127 if slow_path == b'warn':
1128 msg = b"warning: " + msg + b'\n'
1128 msg = b"warning: " + msg + b'\n'
1129 ui.warn(msg)
1129 ui.warn(msg)
1130 if not ui.quiet:
1130 if not ui.quiet:
1131 hint = b'(' + hint + b')\n'
1131 hint = b'(' + hint + b')\n'
1132 ui.warn(hint)
1132 ui.warn(hint)
1133 if slow_path == b'abort':
1133 if slow_path == b'abort':
1134 raise error.Abort(msg, hint=hint)
1134 raise error.Abort(msg, hint=hint)
1135 options[b'persistent-nodemap'] = True
1135 options[b'persistent-nodemap'] = True
1136 if ui.configbool(b'storage', b'revlog.persistent-nodemap.mmap'):
1136 if ui.configbool(b'storage', b'revlog.persistent-nodemap.mmap'):
1137 options[b'persistent-nodemap.mmap'] = True
1137 options[b'persistent-nodemap.mmap'] = True
1138 if ui.configbool(b'devel', b'persistent-nodemap'):
1138 if ui.configbool(b'devel', b'persistent-nodemap'):
1139 options[b'devel-force-nodemap'] = True
1139 options[b'devel-force-nodemap'] = True
1140
1140
1141 return options
1141 return options
1142
1142
1143
1143
1144 def makemain(**kwargs):
1144 def makemain(**kwargs):
1145 """Produce a type conforming to ``ilocalrepositorymain``."""
1145 """Produce a type conforming to ``ilocalrepositorymain``."""
1146 return localrepository
1146 return localrepository
1147
1147
1148
1148
1149 @interfaceutil.implementer(repository.ilocalrepositoryfilestorage)
1149 @interfaceutil.implementer(repository.ilocalrepositoryfilestorage)
1150 class revlogfilestorage(object):
1150 class revlogfilestorage(object):
1151 """File storage when using revlogs."""
1151 """File storage when using revlogs."""
1152
1152
1153 def file(self, path):
1153 def file(self, path):
1154 if path.startswith(b'/'):
1154 if path.startswith(b'/'):
1155 path = path[1:]
1155 path = path[1:]
1156
1156
1157 return filelog.filelog(self.svfs, path)
1157 return filelog.filelog(self.svfs, path)
1158
1158
1159
1159
1160 @interfaceutil.implementer(repository.ilocalrepositoryfilestorage)
1160 @interfaceutil.implementer(repository.ilocalrepositoryfilestorage)
1161 class revlognarrowfilestorage(object):
1161 class revlognarrowfilestorage(object):
1162 """File storage when using revlogs and narrow files."""
1162 """File storage when using revlogs and narrow files."""
1163
1163
1164 def file(self, path):
1164 def file(self, path):
1165 if path.startswith(b'/'):
1165 if path.startswith(b'/'):
1166 path = path[1:]
1166 path = path[1:]
1167
1167
1168 return filelog.narrowfilelog(self.svfs, path, self._storenarrowmatch)
1168 return filelog.narrowfilelog(self.svfs, path, self._storenarrowmatch)
1169
1169
1170
1170
1171 def makefilestorage(requirements, features, **kwargs):
1171 def makefilestorage(requirements, features, **kwargs):
1172 """Produce a type conforming to ``ilocalrepositoryfilestorage``."""
1172 """Produce a type conforming to ``ilocalrepositoryfilestorage``."""
1173 features.add(repository.REPO_FEATURE_REVLOG_FILE_STORAGE)
1173 features.add(repository.REPO_FEATURE_REVLOG_FILE_STORAGE)
1174 features.add(repository.REPO_FEATURE_STREAM_CLONE)
1174 features.add(repository.REPO_FEATURE_STREAM_CLONE)
1175
1175
1176 if requirementsmod.NARROW_REQUIREMENT in requirements:
1176 if requirementsmod.NARROW_REQUIREMENT in requirements:
1177 return revlognarrowfilestorage
1177 return revlognarrowfilestorage
1178 else:
1178 else:
1179 return revlogfilestorage
1179 return revlogfilestorage
1180
1180
1181
1181
1182 # List of repository interfaces and factory functions for them. Each
1182 # List of repository interfaces and factory functions for them. Each
1183 # will be called in order during ``makelocalrepository()`` to iteratively
1183 # will be called in order during ``makelocalrepository()`` to iteratively
1184 # derive the final type for a local repository instance. We capture the
1184 # derive the final type for a local repository instance. We capture the
1185 # function as a lambda so we don't hold a reference and the module-level
1185 # function as a lambda so we don't hold a reference and the module-level
1186 # functions can be wrapped.
1186 # functions can be wrapped.
1187 REPO_INTERFACES = [
1187 REPO_INTERFACES = [
1188 (repository.ilocalrepositorymain, lambda: makemain),
1188 (repository.ilocalrepositorymain, lambda: makemain),
1189 (repository.ilocalrepositoryfilestorage, lambda: makefilestorage),
1189 (repository.ilocalrepositoryfilestorage, lambda: makefilestorage),
1190 ]
1190 ]
1191
1191
1192
1192
1193 @interfaceutil.implementer(repository.ilocalrepositorymain)
1193 @interfaceutil.implementer(repository.ilocalrepositorymain)
1194 class localrepository(object):
1194 class localrepository(object):
1195 """Main class for representing local repositories.
1195 """Main class for representing local repositories.
1196
1196
1197 All local repositories are instances of this class.
1197 All local repositories are instances of this class.
1198
1198
1199 Constructed on its own, instances of this class are not usable as
1199 Constructed on its own, instances of this class are not usable as
1200 repository objects. To obtain a usable repository object, call
1200 repository objects. To obtain a usable repository object, call
1201 ``hg.repository()``, ``localrepo.instance()``, or
1201 ``hg.repository()``, ``localrepo.instance()``, or
1202 ``localrepo.makelocalrepository()``. The latter is the lowest-level.
1202 ``localrepo.makelocalrepository()``. The latter is the lowest-level.
1203 ``instance()`` adds support for creating new repositories.
1203 ``instance()`` adds support for creating new repositories.
1204 ``hg.repository()`` adds more extension integration, including calling
1204 ``hg.repository()`` adds more extension integration, including calling
1205 ``reposetup()``. Generally speaking, ``hg.repository()`` should be
1205 ``reposetup()``. Generally speaking, ``hg.repository()`` should be
1206 used.
1206 used.
1207 """
1207 """
1208
1208
1209 # obsolete experimental requirements:
1209 # obsolete experimental requirements:
1210 # - manifestv2: An experimental new manifest format that allowed
1210 # - manifestv2: An experimental new manifest format that allowed
1211 # for stem compression of long paths. Experiment ended up not
1211 # for stem compression of long paths. Experiment ended up not
1212 # being successful (repository sizes went up due to worse delta
1212 # being successful (repository sizes went up due to worse delta
1213 # chains), and the code was deleted in 4.6.
1213 # chains), and the code was deleted in 4.6.
1214 supportedformats = {
1214 supportedformats = {
1215 requirementsmod.REVLOGV1_REQUIREMENT,
1215 requirementsmod.REVLOGV1_REQUIREMENT,
1216 requirementsmod.GENERALDELTA_REQUIREMENT,
1216 requirementsmod.GENERALDELTA_REQUIREMENT,
1217 requirementsmod.TREEMANIFEST_REQUIREMENT,
1217 requirementsmod.TREEMANIFEST_REQUIREMENT,
1218 requirementsmod.COPIESSDC_REQUIREMENT,
1218 requirementsmod.COPIESSDC_REQUIREMENT,
1219 requirementsmod.REVLOGV2_REQUIREMENT,
1219 requirementsmod.REVLOGV2_REQUIREMENT,
1220 requirementsmod.SIDEDATA_REQUIREMENT,
1220 requirementsmod.SIDEDATA_REQUIREMENT,
1221 requirementsmod.SPARSEREVLOG_REQUIREMENT,
1221 requirementsmod.SPARSEREVLOG_REQUIREMENT,
1222 requirementsmod.NODEMAP_REQUIREMENT,
1222 requirementsmod.NODEMAP_REQUIREMENT,
1223 bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT,
1223 bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT,
1224 requirementsmod.SHARESAFE_REQUIREMENT,
1224 requirementsmod.SHARESAFE_REQUIREMENT,
1225 }
1225 }
1226 _basesupported = supportedformats | {
1226 _basesupported = supportedformats | {
1227 requirementsmod.STORE_REQUIREMENT,
1227 requirementsmod.STORE_REQUIREMENT,
1228 requirementsmod.FNCACHE_REQUIREMENT,
1228 requirementsmod.FNCACHE_REQUIREMENT,
1229 requirementsmod.SHARED_REQUIREMENT,
1229 requirementsmod.SHARED_REQUIREMENT,
1230 requirementsmod.RELATIVE_SHARED_REQUIREMENT,
1230 requirementsmod.RELATIVE_SHARED_REQUIREMENT,
1231 requirementsmod.DOTENCODE_REQUIREMENT,
1231 requirementsmod.DOTENCODE_REQUIREMENT,
1232 requirementsmod.SPARSE_REQUIREMENT,
1232 requirementsmod.SPARSE_REQUIREMENT,
1233 requirementsmod.INTERNAL_PHASE_REQUIREMENT,
1233 requirementsmod.INTERNAL_PHASE_REQUIREMENT,
1234 }
1234 }
1235
1235
1236 # list of prefix for file which can be written without 'wlock'
1236 # list of prefix for file which can be written without 'wlock'
1237 # Extensions should extend this list when needed
1237 # Extensions should extend this list when needed
1238 _wlockfreeprefix = {
1238 _wlockfreeprefix = {
1239 # We migh consider requiring 'wlock' for the next
1239 # We migh consider requiring 'wlock' for the next
1240 # two, but pretty much all the existing code assume
1240 # two, but pretty much all the existing code assume
1241 # wlock is not needed so we keep them excluded for
1241 # wlock is not needed so we keep them excluded for
1242 # now.
1242 # now.
1243 b'hgrc',
1243 b'hgrc',
1244 b'requires',
1244 b'requires',
1245 # XXX cache is a complicatged business someone
1245 # XXX cache is a complicatged business someone
1246 # should investigate this in depth at some point
1246 # should investigate this in depth at some point
1247 b'cache/',
1247 b'cache/',
1248 # XXX shouldn't be dirstate covered by the wlock?
1248 # XXX shouldn't be dirstate covered by the wlock?
1249 b'dirstate',
1249 b'dirstate',
1250 # XXX bisect was still a bit too messy at the time
1250 # XXX bisect was still a bit too messy at the time
1251 # this changeset was introduced. Someone should fix
1251 # this changeset was introduced. Someone should fix
1252 # the remainig bit and drop this line
1252 # the remainig bit and drop this line
1253 b'bisect.state',
1253 b'bisect.state',
1254 }
1254 }
1255
1255
1256 def __init__(
1256 def __init__(
1257 self,
1257 self,
1258 baseui,
1258 baseui,
1259 ui,
1259 ui,
1260 origroot,
1260 origroot,
1261 wdirvfs,
1261 wdirvfs,
1262 hgvfs,
1262 hgvfs,
1263 requirements,
1263 requirements,
1264 supportedrequirements,
1264 supportedrequirements,
1265 sharedpath,
1265 sharedpath,
1266 store,
1266 store,
1267 cachevfs,
1267 cachevfs,
1268 wcachevfs,
1268 wcachevfs,
1269 features,
1269 features,
1270 intents=None,
1270 intents=None,
1271 ):
1271 ):
1272 """Create a new local repository instance.
1272 """Create a new local repository instance.
1273
1273
1274 Most callers should use ``hg.repository()``, ``localrepo.instance()``,
1274 Most callers should use ``hg.repository()``, ``localrepo.instance()``,
1275 or ``localrepo.makelocalrepository()`` for obtaining a new repository
1275 or ``localrepo.makelocalrepository()`` for obtaining a new repository
1276 object.
1276 object.
1277
1277
1278 Arguments:
1278 Arguments:
1279
1279
1280 baseui
1280 baseui
1281 ``ui.ui`` instance that ``ui`` argument was based off of.
1281 ``ui.ui`` instance that ``ui`` argument was based off of.
1282
1282
1283 ui
1283 ui
1284 ``ui.ui`` instance for use by the repository.
1284 ``ui.ui`` instance for use by the repository.
1285
1285
1286 origroot
1286 origroot
1287 ``bytes`` path to working directory root of this repository.
1287 ``bytes`` path to working directory root of this repository.
1288
1288
1289 wdirvfs
1289 wdirvfs
1290 ``vfs.vfs`` rooted at the working directory.
1290 ``vfs.vfs`` rooted at the working directory.
1291
1291
1292 hgvfs
1292 hgvfs
1293 ``vfs.vfs`` rooted at .hg/
1293 ``vfs.vfs`` rooted at .hg/
1294
1294
1295 requirements
1295 requirements
1296 ``set`` of bytestrings representing repository opening requirements.
1296 ``set`` of bytestrings representing repository opening requirements.
1297
1297
1298 supportedrequirements
1298 supportedrequirements
1299 ``set`` of bytestrings representing repository requirements that we
1299 ``set`` of bytestrings representing repository requirements that we
1300 know how to open. May be a supetset of ``requirements``.
1300 know how to open. May be a supetset of ``requirements``.
1301
1301
1302 sharedpath
1302 sharedpath
1303 ``bytes`` Defining path to storage base directory. Points to a
1303 ``bytes`` Defining path to storage base directory. Points to a
1304 ``.hg/`` directory somewhere.
1304 ``.hg/`` directory somewhere.
1305
1305
1306 store
1306 store
1307 ``store.basicstore`` (or derived) instance providing access to
1307 ``store.basicstore`` (or derived) instance providing access to
1308 versioned storage.
1308 versioned storage.
1309
1309
1310 cachevfs
1310 cachevfs
1311 ``vfs.vfs`` used for cache files.
1311 ``vfs.vfs`` used for cache files.
1312
1312
1313 wcachevfs
1313 wcachevfs
1314 ``vfs.vfs`` used for cache files related to the working copy.
1314 ``vfs.vfs`` used for cache files related to the working copy.
1315
1315
1316 features
1316 features
1317 ``set`` of bytestrings defining features/capabilities of this
1317 ``set`` of bytestrings defining features/capabilities of this
1318 instance.
1318 instance.
1319
1319
1320 intents
1320 intents
1321 ``set`` of system strings indicating what this repo will be used
1321 ``set`` of system strings indicating what this repo will be used
1322 for.
1322 for.
1323 """
1323 """
1324 self.baseui = baseui
1324 self.baseui = baseui
1325 self.ui = ui
1325 self.ui = ui
1326 self.origroot = origroot
1326 self.origroot = origroot
1327 # vfs rooted at working directory.
1327 # vfs rooted at working directory.
1328 self.wvfs = wdirvfs
1328 self.wvfs = wdirvfs
1329 self.root = wdirvfs.base
1329 self.root = wdirvfs.base
1330 # vfs rooted at .hg/. Used to access most non-store paths.
1330 # vfs rooted at .hg/. Used to access most non-store paths.
1331 self.vfs = hgvfs
1331 self.vfs = hgvfs
1332 self.path = hgvfs.base
1332 self.path = hgvfs.base
1333 self.requirements = requirements
1333 self.requirements = requirements
1334 self.nodeconstants = sha1nodeconstants
1334 self.nodeconstants = sha1nodeconstants
1335 self.nullid = self.nodeconstants.nullid
1335 self.nullid = self.nodeconstants.nullid
1336 self.supported = supportedrequirements
1336 self.supported = supportedrequirements
1337 self.sharedpath = sharedpath
1337 self.sharedpath = sharedpath
1338 self.store = store
1338 self.store = store
1339 self.cachevfs = cachevfs
1339 self.cachevfs = cachevfs
1340 self.wcachevfs = wcachevfs
1340 self.wcachevfs = wcachevfs
1341 self.features = features
1341 self.features = features
1342
1342
1343 self.filtername = None
1343 self.filtername = None
1344
1344
1345 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
1345 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
1346 b'devel', b'check-locks'
1346 b'devel', b'check-locks'
1347 ):
1347 ):
1348 self.vfs.audit = self._getvfsward(self.vfs.audit)
1348 self.vfs.audit = self._getvfsward(self.vfs.audit)
1349 # A list of callback to shape the phase if no data were found.
1349 # A list of callback to shape the phase if no data were found.
1350 # Callback are in the form: func(repo, roots) --> processed root.
1350 # Callback are in the form: func(repo, roots) --> processed root.
1351 # This list it to be filled by extension during repo setup
1351 # This list it to be filled by extension during repo setup
1352 self._phasedefaults = []
1352 self._phasedefaults = []
1353
1353
1354 color.setup(self.ui)
1354 color.setup(self.ui)
1355
1355
1356 self.spath = self.store.path
1356 self.spath = self.store.path
1357 self.svfs = self.store.vfs
1357 self.svfs = self.store.vfs
1358 self.sjoin = self.store.join
1358 self.sjoin = self.store.join
1359 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
1359 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
1360 b'devel', b'check-locks'
1360 b'devel', b'check-locks'
1361 ):
1361 ):
1362 if util.safehasattr(self.svfs, b'vfs'): # this is filtervfs
1362 if util.safehasattr(self.svfs, b'vfs'): # this is filtervfs
1363 self.svfs.vfs.audit = self._getsvfsward(self.svfs.vfs.audit)
1363 self.svfs.vfs.audit = self._getsvfsward(self.svfs.vfs.audit)
1364 else: # standard vfs
1364 else: # standard vfs
1365 self.svfs.audit = self._getsvfsward(self.svfs.audit)
1365 self.svfs.audit = self._getsvfsward(self.svfs.audit)
1366
1366
1367 self._dirstatevalidatewarned = False
1367 self._dirstatevalidatewarned = False
1368
1368
1369 self._branchcaches = branchmap.BranchMapCache()
1369 self._branchcaches = branchmap.BranchMapCache()
1370 self._revbranchcache = None
1370 self._revbranchcache = None
1371 self._filterpats = {}
1371 self._filterpats = {}
1372 self._datafilters = {}
1372 self._datafilters = {}
1373 self._transref = self._lockref = self._wlockref = None
1373 self._transref = self._lockref = self._wlockref = None
1374
1374
1375 # A cache for various files under .hg/ that tracks file changes,
1375 # A cache for various files under .hg/ that tracks file changes,
1376 # (used by the filecache decorator)
1376 # (used by the filecache decorator)
1377 #
1377 #
1378 # Maps a property name to its util.filecacheentry
1378 # Maps a property name to its util.filecacheentry
1379 self._filecache = {}
1379 self._filecache = {}
1380
1380
1381 # hold sets of revision to be filtered
1381 # hold sets of revision to be filtered
1382 # should be cleared when something might have changed the filter value:
1382 # should be cleared when something might have changed the filter value:
1383 # - new changesets,
1383 # - new changesets,
1384 # - phase change,
1384 # - phase change,
1385 # - new obsolescence marker,
1385 # - new obsolescence marker,
1386 # - working directory parent change,
1386 # - working directory parent change,
1387 # - bookmark changes
1387 # - bookmark changes
1388 self.filteredrevcache = {}
1388 self.filteredrevcache = {}
1389
1389
1390 # post-dirstate-status hooks
1390 # post-dirstate-status hooks
1391 self._postdsstatus = []
1391 self._postdsstatus = []
1392
1392
1393 # generic mapping between names and nodes
1393 # generic mapping between names and nodes
1394 self.names = namespaces.namespaces()
1394 self.names = namespaces.namespaces()
1395
1395
1396 # Key to signature value.
1396 # Key to signature value.
1397 self._sparsesignaturecache = {}
1397 self._sparsesignaturecache = {}
1398 # Signature to cached matcher instance.
1398 # Signature to cached matcher instance.
1399 self._sparsematchercache = {}
1399 self._sparsematchercache = {}
1400
1400
1401 self._extrafilterid = repoview.extrafilter(ui)
1401 self._extrafilterid = repoview.extrafilter(ui)
1402
1402
1403 self.filecopiesmode = None
1403 self.filecopiesmode = None
1404 if requirementsmod.COPIESSDC_REQUIREMENT in self.requirements:
1404 if requirementsmod.COPIESSDC_REQUIREMENT in self.requirements:
1405 self.filecopiesmode = b'changeset-sidedata'
1405 self.filecopiesmode = b'changeset-sidedata'
1406
1406
1407 self._wanted_sidedata = set()
1407 self._wanted_sidedata = set()
1408 self._sidedata_computers = {}
1408 self._sidedata_computers = {}
1409 metadatamod.set_sidedata_spec_for_repo(self)
1409 metadatamod.set_sidedata_spec_for_repo(self)
1410
1410
1411 def _getvfsward(self, origfunc):
1411 def _getvfsward(self, origfunc):
1412 """build a ward for self.vfs"""
1412 """build a ward for self.vfs"""
1413 rref = weakref.ref(self)
1413 rref = weakref.ref(self)
1414
1414
1415 def checkvfs(path, mode=None):
1415 def checkvfs(path, mode=None):
1416 ret = origfunc(path, mode=mode)
1416 ret = origfunc(path, mode=mode)
1417 repo = rref()
1417 repo = rref()
1418 if (
1418 if (
1419 repo is None
1419 repo is None
1420 or not util.safehasattr(repo, b'_wlockref')
1420 or not util.safehasattr(repo, b'_wlockref')
1421 or not util.safehasattr(repo, b'_lockref')
1421 or not util.safehasattr(repo, b'_lockref')
1422 ):
1422 ):
1423 return
1423 return
1424 if mode in (None, b'r', b'rb'):
1424 if mode in (None, b'r', b'rb'):
1425 return
1425 return
1426 if path.startswith(repo.path):
1426 if path.startswith(repo.path):
1427 # truncate name relative to the repository (.hg)
1427 # truncate name relative to the repository (.hg)
1428 path = path[len(repo.path) + 1 :]
1428 path = path[len(repo.path) + 1 :]
1429 if path.startswith(b'cache/'):
1429 if path.startswith(b'cache/'):
1430 msg = b'accessing cache with vfs instead of cachevfs: "%s"'
1430 msg = b'accessing cache with vfs instead of cachevfs: "%s"'
1431 repo.ui.develwarn(msg % path, stacklevel=3, config=b"cache-vfs")
1431 repo.ui.develwarn(msg % path, stacklevel=3, config=b"cache-vfs")
1432 # path prefixes covered by 'lock'
1432 # path prefixes covered by 'lock'
1433 vfs_path_prefixes = (
1433 vfs_path_prefixes = (
1434 b'journal.',
1434 b'journal.',
1435 b'undo.',
1435 b'undo.',
1436 b'strip-backup/',
1436 b'strip-backup/',
1437 b'cache/',
1437 b'cache/',
1438 )
1438 )
1439 if any(path.startswith(prefix) for prefix in vfs_path_prefixes):
1439 if any(path.startswith(prefix) for prefix in vfs_path_prefixes):
1440 if repo._currentlock(repo._lockref) is None:
1440 if repo._currentlock(repo._lockref) is None:
1441 repo.ui.develwarn(
1441 repo.ui.develwarn(
1442 b'write with no lock: "%s"' % path,
1442 b'write with no lock: "%s"' % path,
1443 stacklevel=3,
1443 stacklevel=3,
1444 config=b'check-locks',
1444 config=b'check-locks',
1445 )
1445 )
1446 elif repo._currentlock(repo._wlockref) is None:
1446 elif repo._currentlock(repo._wlockref) is None:
1447 # rest of vfs files are covered by 'wlock'
1447 # rest of vfs files are covered by 'wlock'
1448 #
1448 #
1449 # exclude special files
1449 # exclude special files
1450 for prefix in self._wlockfreeprefix:
1450 for prefix in self._wlockfreeprefix:
1451 if path.startswith(prefix):
1451 if path.startswith(prefix):
1452 return
1452 return
1453 repo.ui.develwarn(
1453 repo.ui.develwarn(
1454 b'write with no wlock: "%s"' % path,
1454 b'write with no wlock: "%s"' % path,
1455 stacklevel=3,
1455 stacklevel=3,
1456 config=b'check-locks',
1456 config=b'check-locks',
1457 )
1457 )
1458 return ret
1458 return ret
1459
1459
1460 return checkvfs
1460 return checkvfs
1461
1461
1462 def _getsvfsward(self, origfunc):
1462 def _getsvfsward(self, origfunc):
1463 """build a ward for self.svfs"""
1463 """build a ward for self.svfs"""
1464 rref = weakref.ref(self)
1464 rref = weakref.ref(self)
1465
1465
1466 def checksvfs(path, mode=None):
1466 def checksvfs(path, mode=None):
1467 ret = origfunc(path, mode=mode)
1467 ret = origfunc(path, mode=mode)
1468 repo = rref()
1468 repo = rref()
1469 if repo is None or not util.safehasattr(repo, b'_lockref'):
1469 if repo is None or not util.safehasattr(repo, b'_lockref'):
1470 return
1470 return
1471 if mode in (None, b'r', b'rb'):
1471 if mode in (None, b'r', b'rb'):
1472 return
1472 return
1473 if path.startswith(repo.sharedpath):
1473 if path.startswith(repo.sharedpath):
1474 # truncate name relative to the repository (.hg)
1474 # truncate name relative to the repository (.hg)
1475 path = path[len(repo.sharedpath) + 1 :]
1475 path = path[len(repo.sharedpath) + 1 :]
1476 if repo._currentlock(repo._lockref) is None:
1476 if repo._currentlock(repo._lockref) is None:
1477 repo.ui.develwarn(
1477 repo.ui.develwarn(
1478 b'write with no lock: "%s"' % path, stacklevel=4
1478 b'write with no lock: "%s"' % path, stacklevel=4
1479 )
1479 )
1480 return ret
1480 return ret
1481
1481
1482 return checksvfs
1482 return checksvfs
1483
1483
1484 def close(self):
1484 def close(self):
1485 self._writecaches()
1485 self._writecaches()
1486
1486
1487 def _writecaches(self):
1487 def _writecaches(self):
1488 if self._revbranchcache:
1488 if self._revbranchcache:
1489 self._revbranchcache.write()
1489 self._revbranchcache.write()
1490
1490
1491 def _restrictcapabilities(self, caps):
1491 def _restrictcapabilities(self, caps):
1492 if self.ui.configbool(b'experimental', b'bundle2-advertise'):
1492 if self.ui.configbool(b'experimental', b'bundle2-advertise'):
1493 caps = set(caps)
1493 caps = set(caps)
1494 capsblob = bundle2.encodecaps(
1494 capsblob = bundle2.encodecaps(
1495 bundle2.getrepocaps(self, role=b'client')
1495 bundle2.getrepocaps(self, role=b'client')
1496 )
1496 )
1497 caps.add(b'bundle2=' + urlreq.quote(capsblob))
1497 caps.add(b'bundle2=' + urlreq.quote(capsblob))
1498 return caps
1498 return caps
1499
1499
1500 # Don't cache auditor/nofsauditor, or you'll end up with reference cycle:
1500 # Don't cache auditor/nofsauditor, or you'll end up with reference cycle:
1501 # self -> auditor -> self._checknested -> self
1501 # self -> auditor -> self._checknested -> self
1502
1502
1503 @property
1503 @property
1504 def auditor(self):
1504 def auditor(self):
1505 # This is only used by context.workingctx.match in order to
1505 # This is only used by context.workingctx.match in order to
1506 # detect files in subrepos.
1506 # detect files in subrepos.
1507 return pathutil.pathauditor(self.root, callback=self._checknested)
1507 return pathutil.pathauditor(self.root, callback=self._checknested)
1508
1508
1509 @property
1509 @property
1510 def nofsauditor(self):
1510 def nofsauditor(self):
1511 # This is only used by context.basectx.match in order to detect
1511 # This is only used by context.basectx.match in order to detect
1512 # files in subrepos.
1512 # files in subrepos.
1513 return pathutil.pathauditor(
1513 return pathutil.pathauditor(
1514 self.root, callback=self._checknested, realfs=False, cached=True
1514 self.root, callback=self._checknested, realfs=False, cached=True
1515 )
1515 )
1516
1516
1517 def _checknested(self, path):
1517 def _checknested(self, path):
1518 """Determine if path is a legal nested repository."""
1518 """Determine if path is a legal nested repository."""
1519 if not path.startswith(self.root):
1519 if not path.startswith(self.root):
1520 return False
1520 return False
1521 subpath = path[len(self.root) + 1 :]
1521 subpath = path[len(self.root) + 1 :]
1522 normsubpath = util.pconvert(subpath)
1522 normsubpath = util.pconvert(subpath)
1523
1523
1524 # XXX: Checking against the current working copy is wrong in
1524 # XXX: Checking against the current working copy is wrong in
1525 # the sense that it can reject things like
1525 # the sense that it can reject things like
1526 #
1526 #
1527 # $ hg cat -r 10 sub/x.txt
1527 # $ hg cat -r 10 sub/x.txt
1528 #
1528 #
1529 # if sub/ is no longer a subrepository in the working copy
1529 # if sub/ is no longer a subrepository in the working copy
1530 # parent revision.
1530 # parent revision.
1531 #
1531 #
1532 # However, it can of course also allow things that would have
1532 # However, it can of course also allow things that would have
1533 # been rejected before, such as the above cat command if sub/
1533 # been rejected before, such as the above cat command if sub/
1534 # is a subrepository now, but was a normal directory before.
1534 # is a subrepository now, but was a normal directory before.
1535 # The old path auditor would have rejected by mistake since it
1535 # The old path auditor would have rejected by mistake since it
1536 # panics when it sees sub/.hg/.
1536 # panics when it sees sub/.hg/.
1537 #
1537 #
1538 # All in all, checking against the working copy seems sensible
1538 # All in all, checking against the working copy seems sensible
1539 # since we want to prevent access to nested repositories on
1539 # since we want to prevent access to nested repositories on
1540 # the filesystem *now*.
1540 # the filesystem *now*.
1541 ctx = self[None]
1541 ctx = self[None]
1542 parts = util.splitpath(subpath)
1542 parts = util.splitpath(subpath)
1543 while parts:
1543 while parts:
1544 prefix = b'/'.join(parts)
1544 prefix = b'/'.join(parts)
1545 if prefix in ctx.substate:
1545 if prefix in ctx.substate:
1546 if prefix == normsubpath:
1546 if prefix == normsubpath:
1547 return True
1547 return True
1548 else:
1548 else:
1549 sub = ctx.sub(prefix)
1549 sub = ctx.sub(prefix)
1550 return sub.checknested(subpath[len(prefix) + 1 :])
1550 return sub.checknested(subpath[len(prefix) + 1 :])
1551 else:
1551 else:
1552 parts.pop()
1552 parts.pop()
1553 return False
1553 return False
1554
1554
1555 def peer(self):
1555 def peer(self):
1556 return localpeer(self) # not cached to avoid reference cycle
1556 return localpeer(self) # not cached to avoid reference cycle
1557
1557
1558 def unfiltered(self):
1558 def unfiltered(self):
1559 """Return unfiltered version of the repository
1559 """Return unfiltered version of the repository
1560
1560
1561 Intended to be overwritten by filtered repo."""
1561 Intended to be overwritten by filtered repo."""
1562 return self
1562 return self
1563
1563
1564 def filtered(self, name, visibilityexceptions=None):
1564 def filtered(self, name, visibilityexceptions=None):
1565 """Return a filtered version of a repository
1565 """Return a filtered version of a repository
1566
1566
1567 The `name` parameter is the identifier of the requested view. This
1567 The `name` parameter is the identifier of the requested view. This
1568 will return a repoview object set "exactly" to the specified view.
1568 will return a repoview object set "exactly" to the specified view.
1569
1569
1570 This function does not apply recursive filtering to a repository. For
1570 This function does not apply recursive filtering to a repository. For
1571 example calling `repo.filtered("served")` will return a repoview using
1571 example calling `repo.filtered("served")` will return a repoview using
1572 the "served" view, regardless of the initial view used by `repo`.
1572 the "served" view, regardless of the initial view used by `repo`.
1573
1573
1574 In other word, there is always only one level of `repoview` "filtering".
1574 In other word, there is always only one level of `repoview` "filtering".
1575 """
1575 """
1576 if self._extrafilterid is not None and b'%' not in name:
1576 if self._extrafilterid is not None and b'%' not in name:
1577 name = name + b'%' + self._extrafilterid
1577 name = name + b'%' + self._extrafilterid
1578
1578
1579 cls = repoview.newtype(self.unfiltered().__class__)
1579 cls = repoview.newtype(self.unfiltered().__class__)
1580 return cls(self, name, visibilityexceptions)
1580 return cls(self, name, visibilityexceptions)
1581
1581
1582 @mixedrepostorecache(
1582 @mixedrepostorecache(
1583 (b'bookmarks', b'plain'),
1583 (b'bookmarks', b'plain'),
1584 (b'bookmarks.current', b'plain'),
1584 (b'bookmarks.current', b'plain'),
1585 (b'bookmarks', b''),
1585 (b'bookmarks', b''),
1586 (b'00changelog.i', b''),
1586 (b'00changelog.i', b''),
1587 )
1587 )
1588 def _bookmarks(self):
1588 def _bookmarks(self):
1589 # Since the multiple files involved in the transaction cannot be
1589 # Since the multiple files involved in the transaction cannot be
1590 # written atomically (with current repository format), there is a race
1590 # written atomically (with current repository format), there is a race
1591 # condition here.
1591 # condition here.
1592 #
1592 #
1593 # 1) changelog content A is read
1593 # 1) changelog content A is read
1594 # 2) outside transaction update changelog to content B
1594 # 2) outside transaction update changelog to content B
1595 # 3) outside transaction update bookmark file referring to content B
1595 # 3) outside transaction update bookmark file referring to content B
1596 # 4) bookmarks file content is read and filtered against changelog-A
1596 # 4) bookmarks file content is read and filtered against changelog-A
1597 #
1597 #
1598 # When this happens, bookmarks against nodes missing from A are dropped.
1598 # When this happens, bookmarks against nodes missing from A are dropped.
1599 #
1599 #
1600 # Having this happening during read is not great, but it become worse
1600 # Having this happening during read is not great, but it become worse
1601 # when this happen during write because the bookmarks to the "unknown"
1601 # when this happen during write because the bookmarks to the "unknown"
1602 # nodes will be dropped for good. However, writes happen within locks.
1602 # nodes will be dropped for good. However, writes happen within locks.
1603 # This locking makes it possible to have a race free consistent read.
1603 # This locking makes it possible to have a race free consistent read.
1604 # For this purpose data read from disc before locking are
1604 # For this purpose data read from disc before locking are
1605 # "invalidated" right after the locks are taken. This invalidations are
1605 # "invalidated" right after the locks are taken. This invalidations are
1606 # "light", the `filecache` mechanism keep the data in memory and will
1606 # "light", the `filecache` mechanism keep the data in memory and will
1607 # reuse them if the underlying files did not changed. Not parsing the
1607 # reuse them if the underlying files did not changed. Not parsing the
1608 # same data multiple times helps performances.
1608 # same data multiple times helps performances.
1609 #
1609 #
1610 # Unfortunately in the case describe above, the files tracked by the
1610 # Unfortunately in the case describe above, the files tracked by the
1611 # bookmarks file cache might not have changed, but the in-memory
1611 # bookmarks file cache might not have changed, but the in-memory
1612 # content is still "wrong" because we used an older changelog content
1612 # content is still "wrong" because we used an older changelog content
1613 # to process the on-disk data. So after locking, the changelog would be
1613 # to process the on-disk data. So after locking, the changelog would be
1614 # refreshed but `_bookmarks` would be preserved.
1614 # refreshed but `_bookmarks` would be preserved.
1615 # Adding `00changelog.i` to the list of tracked file is not
1615 # Adding `00changelog.i` to the list of tracked file is not
1616 # enough, because at the time we build the content for `_bookmarks` in
1616 # enough, because at the time we build the content for `_bookmarks` in
1617 # (4), the changelog file has already diverged from the content used
1617 # (4), the changelog file has already diverged from the content used
1618 # for loading `changelog` in (1)
1618 # for loading `changelog` in (1)
1619 #
1619 #
1620 # To prevent the issue, we force the changelog to be explicitly
1620 # To prevent the issue, we force the changelog to be explicitly
1621 # reloaded while computing `_bookmarks`. The data race can still happen
1621 # reloaded while computing `_bookmarks`. The data race can still happen
1622 # without the lock (with a narrower window), but it would no longer go
1622 # without the lock (with a narrower window), but it would no longer go
1623 # undetected during the lock time refresh.
1623 # undetected during the lock time refresh.
1624 #
1624 #
1625 # The new schedule is as follow
1625 # The new schedule is as follow
1626 #
1626 #
1627 # 1) filecache logic detect that `_bookmarks` needs to be computed
1627 # 1) filecache logic detect that `_bookmarks` needs to be computed
1628 # 2) cachestat for `bookmarks` and `changelog` are captured (for book)
1628 # 2) cachestat for `bookmarks` and `changelog` are captured (for book)
1629 # 3) We force `changelog` filecache to be tested
1629 # 3) We force `changelog` filecache to be tested
1630 # 4) cachestat for `changelog` are captured (for changelog)
1630 # 4) cachestat for `changelog` are captured (for changelog)
1631 # 5) `_bookmarks` is computed and cached
1631 # 5) `_bookmarks` is computed and cached
1632 #
1632 #
1633 # The step in (3) ensure we have a changelog at least as recent as the
1633 # The step in (3) ensure we have a changelog at least as recent as the
1634 # cache stat computed in (1). As a result at locking time:
1634 # cache stat computed in (1). As a result at locking time:
1635 # * if the changelog did not changed since (1) -> we can reuse the data
1635 # * if the changelog did not changed since (1) -> we can reuse the data
1636 # * otherwise -> the bookmarks get refreshed.
1636 # * otherwise -> the bookmarks get refreshed.
1637 self._refreshchangelog()
1637 self._refreshchangelog()
1638 return bookmarks.bmstore(self)
1638 return bookmarks.bmstore(self)
1639
1639
1640 def _refreshchangelog(self):
1640 def _refreshchangelog(self):
1641 """make sure the in memory changelog match the on-disk one"""
1641 """make sure the in memory changelog match the on-disk one"""
1642 if 'changelog' in vars(self) and self.currenttransaction() is None:
1642 if 'changelog' in vars(self) and self.currenttransaction() is None:
1643 del self.changelog
1643 del self.changelog
1644
1644
1645 @property
1645 @property
1646 def _activebookmark(self):
1646 def _activebookmark(self):
1647 return self._bookmarks.active
1647 return self._bookmarks.active
1648
1648
1649 # _phasesets depend on changelog. what we need is to call
1649 # _phasesets depend on changelog. what we need is to call
1650 # _phasecache.invalidate() if '00changelog.i' was changed, but it
1650 # _phasecache.invalidate() if '00changelog.i' was changed, but it
1651 # can't be easily expressed in filecache mechanism.
1651 # can't be easily expressed in filecache mechanism.
1652 @storecache(b'phaseroots', b'00changelog.i')
1652 @storecache(b'phaseroots', b'00changelog.i')
1653 def _phasecache(self):
1653 def _phasecache(self):
1654 return phases.phasecache(self, self._phasedefaults)
1654 return phases.phasecache(self, self._phasedefaults)
1655
1655
1656 @storecache(b'obsstore')
1656 @storecache(b'obsstore')
1657 def obsstore(self):
1657 def obsstore(self):
1658 return obsolete.makestore(self.ui, self)
1658 return obsolete.makestore(self.ui, self)
1659
1659
1660 @storecache(b'00changelog.i')
1660 @storecache(b'00changelog.i')
1661 def changelog(self):
1661 def changelog(self):
1662 # load dirstate before changelog to avoid race see issue6303
1662 # load dirstate before changelog to avoid race see issue6303
1663 self.dirstate.prefetch_parents()
1663 self.dirstate.prefetch_parents()
1664 return self.store.changelog(
1664 return self.store.changelog(
1665 txnutil.mayhavepending(self.root),
1665 txnutil.mayhavepending(self.root),
1666 concurrencychecker=revlogchecker.get_checker(self.ui, b'changelog'),
1666 concurrencychecker=revlogchecker.get_checker(self.ui, b'changelog'),
1667 )
1667 )
1668
1668
1669 @storecache(b'00manifest.i')
1669 @storecache(b'00manifest.i')
1670 def manifestlog(self):
1670 def manifestlog(self):
1671 return self.store.manifestlog(self, self._storenarrowmatch)
1671 return self.store.manifestlog(self, self._storenarrowmatch)
1672
1672
1673 @repofilecache(b'dirstate')
1673 @repofilecache(b'dirstate')
1674 def dirstate(self):
1674 def dirstate(self):
1675 return self._makedirstate()
1675 return self._makedirstate()
1676
1676
1677 def _makedirstate(self):
1677 def _makedirstate(self):
1678 """Extension point for wrapping the dirstate per-repo."""
1678 """Extension point for wrapping the dirstate per-repo."""
1679 sparsematchfn = lambda: sparse.matcher(self)
1679 sparsematchfn = lambda: sparse.matcher(self)
1680
1680
1681 return dirstate.dirstate(
1681 return dirstate.dirstate(
1682 self.vfs,
1682 self.vfs,
1683 self.ui,
1683 self.ui,
1684 self.root,
1684 self.root,
1685 self._dirstatevalidate,
1685 self._dirstatevalidate,
1686 sparsematchfn,
1686 sparsematchfn,
1687 self.nodeconstants,
1687 self.nodeconstants,
1688 )
1688 )
1689
1689
1690 def _dirstatevalidate(self, node):
1690 def _dirstatevalidate(self, node):
1691 try:
1691 try:
1692 self.changelog.rev(node)
1692 self.changelog.rev(node)
1693 return node
1693 return node
1694 except error.LookupError:
1694 except error.LookupError:
1695 if not self._dirstatevalidatewarned:
1695 if not self._dirstatevalidatewarned:
1696 self._dirstatevalidatewarned = True
1696 self._dirstatevalidatewarned = True
1697 self.ui.warn(
1697 self.ui.warn(
1698 _(b"warning: ignoring unknown working parent %s!\n")
1698 _(b"warning: ignoring unknown working parent %s!\n")
1699 % short(node)
1699 % short(node)
1700 )
1700 )
1701 return nullid
1701 return nullid
1702
1702
1703 @storecache(narrowspec.FILENAME)
1703 @storecache(narrowspec.FILENAME)
1704 def narrowpats(self):
1704 def narrowpats(self):
1705 """matcher patterns for this repository's narrowspec
1705 """matcher patterns for this repository's narrowspec
1706
1706
1707 A tuple of (includes, excludes).
1707 A tuple of (includes, excludes).
1708 """
1708 """
1709 return narrowspec.load(self)
1709 return narrowspec.load(self)
1710
1710
1711 @storecache(narrowspec.FILENAME)
1711 @storecache(narrowspec.FILENAME)
1712 def _storenarrowmatch(self):
1712 def _storenarrowmatch(self):
1713 if requirementsmod.NARROW_REQUIREMENT not in self.requirements:
1713 if requirementsmod.NARROW_REQUIREMENT not in self.requirements:
1714 return matchmod.always()
1714 return matchmod.always()
1715 include, exclude = self.narrowpats
1715 include, exclude = self.narrowpats
1716 return narrowspec.match(self.root, include=include, exclude=exclude)
1716 return narrowspec.match(self.root, include=include, exclude=exclude)
1717
1717
1718 @storecache(narrowspec.FILENAME)
1718 @storecache(narrowspec.FILENAME)
1719 def _narrowmatch(self):
1719 def _narrowmatch(self):
1720 if requirementsmod.NARROW_REQUIREMENT not in self.requirements:
1720 if requirementsmod.NARROW_REQUIREMENT not in self.requirements:
1721 return matchmod.always()
1721 return matchmod.always()
1722 narrowspec.checkworkingcopynarrowspec(self)
1722 narrowspec.checkworkingcopynarrowspec(self)
1723 include, exclude = self.narrowpats
1723 include, exclude = self.narrowpats
1724 return narrowspec.match(self.root, include=include, exclude=exclude)
1724 return narrowspec.match(self.root, include=include, exclude=exclude)
1725
1725
1726 def narrowmatch(self, match=None, includeexact=False):
1726 def narrowmatch(self, match=None, includeexact=False):
1727 """matcher corresponding the the repo's narrowspec
1727 """matcher corresponding the the repo's narrowspec
1728
1728
1729 If `match` is given, then that will be intersected with the narrow
1729 If `match` is given, then that will be intersected with the narrow
1730 matcher.
1730 matcher.
1731
1731
1732 If `includeexact` is True, then any exact matches from `match` will
1732 If `includeexact` is True, then any exact matches from `match` will
1733 be included even if they're outside the narrowspec.
1733 be included even if they're outside the narrowspec.
1734 """
1734 """
1735 if match:
1735 if match:
1736 if includeexact and not self._narrowmatch.always():
1736 if includeexact and not self._narrowmatch.always():
1737 # do not exclude explicitly-specified paths so that they can
1737 # do not exclude explicitly-specified paths so that they can
1738 # be warned later on
1738 # be warned later on
1739 em = matchmod.exact(match.files())
1739 em = matchmod.exact(match.files())
1740 nm = matchmod.unionmatcher([self._narrowmatch, em])
1740 nm = matchmod.unionmatcher([self._narrowmatch, em])
1741 return matchmod.intersectmatchers(match, nm)
1741 return matchmod.intersectmatchers(match, nm)
1742 return matchmod.intersectmatchers(match, self._narrowmatch)
1742 return matchmod.intersectmatchers(match, self._narrowmatch)
1743 return self._narrowmatch
1743 return self._narrowmatch
1744
1744
1745 def setnarrowpats(self, newincludes, newexcludes):
1745 def setnarrowpats(self, newincludes, newexcludes):
1746 narrowspec.save(self, newincludes, newexcludes)
1746 narrowspec.save(self, newincludes, newexcludes)
1747 self.invalidate(clearfilecache=True)
1747 self.invalidate(clearfilecache=True)
1748
1748
1749 @unfilteredpropertycache
1749 @unfilteredpropertycache
1750 def _quick_access_changeid_null(self):
1750 def _quick_access_changeid_null(self):
1751 return {
1751 return {
1752 b'null': (nullrev, nullid),
1752 b'null': (nullrev, nullid),
1753 nullrev: (nullrev, nullid),
1753 nullrev: (nullrev, nullid),
1754 nullid: (nullrev, nullid),
1754 nullid: (nullrev, nullid),
1755 }
1755 }
1756
1756
1757 @unfilteredpropertycache
1757 @unfilteredpropertycache
1758 def _quick_access_changeid_wc(self):
1758 def _quick_access_changeid_wc(self):
1759 # also fast path access to the working copy parents
1759 # also fast path access to the working copy parents
1760 # however, only do it for filter that ensure wc is visible.
1760 # however, only do it for filter that ensure wc is visible.
1761 quick = self._quick_access_changeid_null.copy()
1761 quick = self._quick_access_changeid_null.copy()
1762 cl = self.unfiltered().changelog
1762 cl = self.unfiltered().changelog
1763 for node in self.dirstate.parents():
1763 for node in self.dirstate.parents():
1764 if node == nullid:
1764 if node == nullid:
1765 continue
1765 continue
1766 rev = cl.index.get_rev(node)
1766 rev = cl.index.get_rev(node)
1767 if rev is None:
1767 if rev is None:
1768 # unknown working copy parent case:
1768 # unknown working copy parent case:
1769 #
1769 #
1770 # skip the fast path and let higher code deal with it
1770 # skip the fast path and let higher code deal with it
1771 continue
1771 continue
1772 pair = (rev, node)
1772 pair = (rev, node)
1773 quick[rev] = pair
1773 quick[rev] = pair
1774 quick[node] = pair
1774 quick[node] = pair
1775 # also add the parents of the parents
1775 # also add the parents of the parents
1776 for r in cl.parentrevs(rev):
1776 for r in cl.parentrevs(rev):
1777 if r == nullrev:
1777 if r == nullrev:
1778 continue
1778 continue
1779 n = cl.node(r)
1779 n = cl.node(r)
1780 pair = (r, n)
1780 pair = (r, n)
1781 quick[r] = pair
1781 quick[r] = pair
1782 quick[n] = pair
1782 quick[n] = pair
1783 p1node = self.dirstate.p1()
1783 p1node = self.dirstate.p1()
1784 if p1node != nullid:
1784 if p1node != nullid:
1785 quick[b'.'] = quick[p1node]
1785 quick[b'.'] = quick[p1node]
1786 return quick
1786 return quick
1787
1787
1788 @unfilteredmethod
1788 @unfilteredmethod
1789 def _quick_access_changeid_invalidate(self):
1789 def _quick_access_changeid_invalidate(self):
1790 if '_quick_access_changeid_wc' in vars(self):
1790 if '_quick_access_changeid_wc' in vars(self):
1791 del self.__dict__['_quick_access_changeid_wc']
1791 del self.__dict__['_quick_access_changeid_wc']
1792
1792
1793 @property
1793 @property
1794 def _quick_access_changeid(self):
1794 def _quick_access_changeid(self):
1795 """an helper dictionnary for __getitem__ calls
1795 """an helper dictionnary for __getitem__ calls
1796
1796
1797 This contains a list of symbol we can recognise right away without
1797 This contains a list of symbol we can recognise right away without
1798 further processing.
1798 further processing.
1799 """
1799 """
1800 if self.filtername in repoview.filter_has_wc:
1800 if self.filtername in repoview.filter_has_wc:
1801 return self._quick_access_changeid_wc
1801 return self._quick_access_changeid_wc
1802 return self._quick_access_changeid_null
1802 return self._quick_access_changeid_null
1803
1803
1804 def __getitem__(self, changeid):
1804 def __getitem__(self, changeid):
1805 # dealing with special cases
1805 # dealing with special cases
1806 if changeid is None:
1806 if changeid is None:
1807 return context.workingctx(self)
1807 return context.workingctx(self)
1808 if isinstance(changeid, context.basectx):
1808 if isinstance(changeid, context.basectx):
1809 return changeid
1809 return changeid
1810
1810
1811 # dealing with multiple revisions
1811 # dealing with multiple revisions
1812 if isinstance(changeid, slice):
1812 if isinstance(changeid, slice):
1813 # wdirrev isn't contiguous so the slice shouldn't include it
1813 # wdirrev isn't contiguous so the slice shouldn't include it
1814 return [
1814 return [
1815 self[i]
1815 self[i]
1816 for i in pycompat.xrange(*changeid.indices(len(self)))
1816 for i in pycompat.xrange(*changeid.indices(len(self)))
1817 if i not in self.changelog.filteredrevs
1817 if i not in self.changelog.filteredrevs
1818 ]
1818 ]
1819
1819
1820 # dealing with some special values
1820 # dealing with some special values
1821 quick_access = self._quick_access_changeid.get(changeid)
1821 quick_access = self._quick_access_changeid.get(changeid)
1822 if quick_access is not None:
1822 if quick_access is not None:
1823 rev, node = quick_access
1823 rev, node = quick_access
1824 return context.changectx(self, rev, node, maybe_filtered=False)
1824 return context.changectx(self, rev, node, maybe_filtered=False)
1825 if changeid == b'tip':
1825 if changeid == b'tip':
1826 node = self.changelog.tip()
1826 node = self.changelog.tip()
1827 rev = self.changelog.rev(node)
1827 rev = self.changelog.rev(node)
1828 return context.changectx(self, rev, node)
1828 return context.changectx(self, rev, node)
1829
1829
1830 # dealing with arbitrary values
1830 # dealing with arbitrary values
1831 try:
1831 try:
1832 if isinstance(changeid, int):
1832 if isinstance(changeid, int):
1833 node = self.changelog.node(changeid)
1833 node = self.changelog.node(changeid)
1834 rev = changeid
1834 rev = changeid
1835 elif changeid == b'.':
1835 elif changeid == b'.':
1836 # this is a hack to delay/avoid loading obsmarkers
1836 # this is a hack to delay/avoid loading obsmarkers
1837 # when we know that '.' won't be hidden
1837 # when we know that '.' won't be hidden
1838 node = self.dirstate.p1()
1838 node = self.dirstate.p1()
1839 rev = self.unfiltered().changelog.rev(node)
1839 rev = self.unfiltered().changelog.rev(node)
1840 elif len(changeid) == 20:
1840 elif len(changeid) == 20:
1841 try:
1841 try:
1842 node = changeid
1842 node = changeid
1843 rev = self.changelog.rev(changeid)
1843 rev = self.changelog.rev(changeid)
1844 except error.FilteredLookupError:
1844 except error.FilteredLookupError:
1845 changeid = hex(changeid) # for the error message
1845 changeid = hex(changeid) # for the error message
1846 raise
1846 raise
1847 except LookupError:
1847 except LookupError:
1848 # check if it might have come from damaged dirstate
1848 # check if it might have come from damaged dirstate
1849 #
1849 #
1850 # XXX we could avoid the unfiltered if we had a recognizable
1850 # XXX we could avoid the unfiltered if we had a recognizable
1851 # exception for filtered changeset access
1851 # exception for filtered changeset access
1852 if (
1852 if (
1853 self.local()
1853 self.local()
1854 and changeid in self.unfiltered().dirstate.parents()
1854 and changeid in self.unfiltered().dirstate.parents()
1855 ):
1855 ):
1856 msg = _(b"working directory has unknown parent '%s'!")
1856 msg = _(b"working directory has unknown parent '%s'!")
1857 raise error.Abort(msg % short(changeid))
1857 raise error.Abort(msg % short(changeid))
1858 changeid = hex(changeid) # for the error message
1858 changeid = hex(changeid) # for the error message
1859 raise
1859 raise
1860
1860
1861 elif len(changeid) == 40:
1861 elif len(changeid) == 40:
1862 node = bin(changeid)
1862 node = bin(changeid)
1863 rev = self.changelog.rev(node)
1863 rev = self.changelog.rev(node)
1864 else:
1864 else:
1865 raise error.ProgrammingError(
1865 raise error.ProgrammingError(
1866 b"unsupported changeid '%s' of type %s"
1866 b"unsupported changeid '%s' of type %s"
1867 % (changeid, pycompat.bytestr(type(changeid)))
1867 % (changeid, pycompat.bytestr(type(changeid)))
1868 )
1868 )
1869
1869
1870 return context.changectx(self, rev, node)
1870 return context.changectx(self, rev, node)
1871
1871
1872 except (error.FilteredIndexError, error.FilteredLookupError):
1872 except (error.FilteredIndexError, error.FilteredLookupError):
1873 raise error.FilteredRepoLookupError(
1873 raise error.FilteredRepoLookupError(
1874 _(b"filtered revision '%s'") % pycompat.bytestr(changeid)
1874 _(b"filtered revision '%s'") % pycompat.bytestr(changeid)
1875 )
1875 )
1876 except (IndexError, LookupError):
1876 except (IndexError, LookupError):
1877 raise error.RepoLookupError(
1877 raise error.RepoLookupError(
1878 _(b"unknown revision '%s'") % pycompat.bytestr(changeid)
1878 _(b"unknown revision '%s'") % pycompat.bytestr(changeid)
1879 )
1879 )
1880 except error.WdirUnsupported:
1880 except error.WdirUnsupported:
1881 return context.workingctx(self)
1881 return context.workingctx(self)
1882
1882
1883 def __contains__(self, changeid):
1883 def __contains__(self, changeid):
1884 """True if the given changeid exists"""
1884 """True if the given changeid exists"""
1885 try:
1885 try:
1886 self[changeid]
1886 self[changeid]
1887 return True
1887 return True
1888 except error.RepoLookupError:
1888 except error.RepoLookupError:
1889 return False
1889 return False
1890
1890
1891 def __nonzero__(self):
1891 def __nonzero__(self):
1892 return True
1892 return True
1893
1893
1894 __bool__ = __nonzero__
1894 __bool__ = __nonzero__
1895
1895
1896 def __len__(self):
1896 def __len__(self):
1897 # no need to pay the cost of repoview.changelog
1897 # no need to pay the cost of repoview.changelog
1898 unfi = self.unfiltered()
1898 unfi = self.unfiltered()
1899 return len(unfi.changelog)
1899 return len(unfi.changelog)
1900
1900
1901 def __iter__(self):
1901 def __iter__(self):
1902 return iter(self.changelog)
1902 return iter(self.changelog)
1903
1903
1904 def revs(self, expr, *args):
1904 def revs(self, expr, *args):
1905 """Find revisions matching a revset.
1905 """Find revisions matching a revset.
1906
1906
1907 The revset is specified as a string ``expr`` that may contain
1907 The revset is specified as a string ``expr`` that may contain
1908 %-formatting to escape certain types. See ``revsetlang.formatspec``.
1908 %-formatting to escape certain types. See ``revsetlang.formatspec``.
1909
1909
1910 Revset aliases from the configuration are not expanded. To expand
1910 Revset aliases from the configuration are not expanded. To expand
1911 user aliases, consider calling ``scmutil.revrange()`` or
1911 user aliases, consider calling ``scmutil.revrange()`` or
1912 ``repo.anyrevs([expr], user=True)``.
1912 ``repo.anyrevs([expr], user=True)``.
1913
1913
1914 Returns a smartset.abstractsmartset, which is a list-like interface
1914 Returns a smartset.abstractsmartset, which is a list-like interface
1915 that contains integer revisions.
1915 that contains integer revisions.
1916 """
1916 """
1917 tree = revsetlang.spectree(expr, *args)
1917 tree = revsetlang.spectree(expr, *args)
1918 return revset.makematcher(tree)(self)
1918 return revset.makematcher(tree)(self)
1919
1919
1920 def set(self, expr, *args):
1920 def set(self, expr, *args):
1921 """Find revisions matching a revset and emit changectx instances.
1921 """Find revisions matching a revset and emit changectx instances.
1922
1922
1923 This is a convenience wrapper around ``revs()`` that iterates the
1923 This is a convenience wrapper around ``revs()`` that iterates the
1924 result and is a generator of changectx instances.
1924 result and is a generator of changectx instances.
1925
1925
1926 Revset aliases from the configuration are not expanded. To expand
1926 Revset aliases from the configuration are not expanded. To expand
1927 user aliases, consider calling ``scmutil.revrange()``.
1927 user aliases, consider calling ``scmutil.revrange()``.
1928 """
1928 """
1929 for r in self.revs(expr, *args):
1929 for r in self.revs(expr, *args):
1930 yield self[r]
1930 yield self[r]
1931
1931
1932 def anyrevs(self, specs, user=False, localalias=None):
1932 def anyrevs(self, specs, user=False, localalias=None):
1933 """Find revisions matching one of the given revsets.
1933 """Find revisions matching one of the given revsets.
1934
1934
1935 Revset aliases from the configuration are not expanded by default. To
1935 Revset aliases from the configuration are not expanded by default. To
1936 expand user aliases, specify ``user=True``. To provide some local
1936 expand user aliases, specify ``user=True``. To provide some local
1937 definitions overriding user aliases, set ``localalias`` to
1937 definitions overriding user aliases, set ``localalias`` to
1938 ``{name: definitionstring}``.
1938 ``{name: definitionstring}``.
1939 """
1939 """
1940 if specs == [b'null']:
1940 if specs == [b'null']:
1941 return revset.baseset([nullrev])
1941 return revset.baseset([nullrev])
1942 if specs == [b'.']:
1942 if specs == [b'.']:
1943 quick_data = self._quick_access_changeid.get(b'.')
1943 quick_data = self._quick_access_changeid.get(b'.')
1944 if quick_data is not None:
1944 if quick_data is not None:
1945 return revset.baseset([quick_data[0]])
1945 return revset.baseset([quick_data[0]])
1946 if user:
1946 if user:
1947 m = revset.matchany(
1947 m = revset.matchany(
1948 self.ui,
1948 self.ui,
1949 specs,
1949 specs,
1950 lookup=revset.lookupfn(self),
1950 lookup=revset.lookupfn(self),
1951 localalias=localalias,
1951 localalias=localalias,
1952 )
1952 )
1953 else:
1953 else:
1954 m = revset.matchany(None, specs, localalias=localalias)
1954 m = revset.matchany(None, specs, localalias=localalias)
1955 return m(self)
1955 return m(self)
1956
1956
1957 def url(self):
1957 def url(self):
1958 return b'file:' + self.root
1958 return b'file:' + self.root
1959
1959
1960 def hook(self, name, throw=False, **args):
1960 def hook(self, name, throw=False, **args):
1961 """Call a hook, passing this repo instance.
1961 """Call a hook, passing this repo instance.
1962
1962
1963 This a convenience method to aid invoking hooks. Extensions likely
1963 This a convenience method to aid invoking hooks. Extensions likely
1964 won't call this unless they have registered a custom hook or are
1964 won't call this unless they have registered a custom hook or are
1965 replacing code that is expected to call a hook.
1965 replacing code that is expected to call a hook.
1966 """
1966 """
1967 return hook.hook(self.ui, self, name, throw, **args)
1967 return hook.hook(self.ui, self, name, throw, **args)
1968
1968
1969 @filteredpropertycache
1969 @filteredpropertycache
1970 def _tagscache(self):
1970 def _tagscache(self):
1971 """Returns a tagscache object that contains various tags related
1971 """Returns a tagscache object that contains various tags related
1972 caches."""
1972 caches."""
1973
1973
1974 # This simplifies its cache management by having one decorated
1974 # This simplifies its cache management by having one decorated
1975 # function (this one) and the rest simply fetch things from it.
1975 # function (this one) and the rest simply fetch things from it.
1976 class tagscache(object):
1976 class tagscache(object):
1977 def __init__(self):
1977 def __init__(self):
1978 # These two define the set of tags for this repository. tags
1978 # These two define the set of tags for this repository. tags
1979 # maps tag name to node; tagtypes maps tag name to 'global' or
1979 # maps tag name to node; tagtypes maps tag name to 'global' or
1980 # 'local'. (Global tags are defined by .hgtags across all
1980 # 'local'. (Global tags are defined by .hgtags across all
1981 # heads, and local tags are defined in .hg/localtags.)
1981 # heads, and local tags are defined in .hg/localtags.)
1982 # They constitute the in-memory cache of tags.
1982 # They constitute the in-memory cache of tags.
1983 self.tags = self.tagtypes = None
1983 self.tags = self.tagtypes = None
1984
1984
1985 self.nodetagscache = self.tagslist = None
1985 self.nodetagscache = self.tagslist = None
1986
1986
1987 cache = tagscache()
1987 cache = tagscache()
1988 cache.tags, cache.tagtypes = self._findtags()
1988 cache.tags, cache.tagtypes = self._findtags()
1989
1989
1990 return cache
1990 return cache
1991
1991
1992 def tags(self):
1992 def tags(self):
1993 '''return a mapping of tag to node'''
1993 '''return a mapping of tag to node'''
1994 t = {}
1994 t = {}
1995 if self.changelog.filteredrevs:
1995 if self.changelog.filteredrevs:
1996 tags, tt = self._findtags()
1996 tags, tt = self._findtags()
1997 else:
1997 else:
1998 tags = self._tagscache.tags
1998 tags = self._tagscache.tags
1999 rev = self.changelog.rev
1999 rev = self.changelog.rev
2000 for k, v in pycompat.iteritems(tags):
2000 for k, v in pycompat.iteritems(tags):
2001 try:
2001 try:
2002 # ignore tags to unknown nodes
2002 # ignore tags to unknown nodes
2003 rev(v)
2003 rev(v)
2004 t[k] = v
2004 t[k] = v
2005 except (error.LookupError, ValueError):
2005 except (error.LookupError, ValueError):
2006 pass
2006 pass
2007 return t
2007 return t
2008
2008
2009 def _findtags(self):
2009 def _findtags(self):
2010 """Do the hard work of finding tags. Return a pair of dicts
2010 """Do the hard work of finding tags. Return a pair of dicts
2011 (tags, tagtypes) where tags maps tag name to node, and tagtypes
2011 (tags, tagtypes) where tags maps tag name to node, and tagtypes
2012 maps tag name to a string like \'global\' or \'local\'.
2012 maps tag name to a string like \'global\' or \'local\'.
2013 Subclasses or extensions are free to add their own tags, but
2013 Subclasses or extensions are free to add their own tags, but
2014 should be aware that the returned dicts will be retained for the
2014 should be aware that the returned dicts will be retained for the
2015 duration of the localrepo object."""
2015 duration of the localrepo object."""
2016
2016
2017 # XXX what tagtype should subclasses/extensions use? Currently
2017 # XXX what tagtype should subclasses/extensions use? Currently
2018 # mq and bookmarks add tags, but do not set the tagtype at all.
2018 # mq and bookmarks add tags, but do not set the tagtype at all.
2019 # Should each extension invent its own tag type? Should there
2019 # Should each extension invent its own tag type? Should there
2020 # be one tagtype for all such "virtual" tags? Or is the status
2020 # be one tagtype for all such "virtual" tags? Or is the status
2021 # quo fine?
2021 # quo fine?
2022
2022
2023 # map tag name to (node, hist)
2023 # map tag name to (node, hist)
2024 alltags = tagsmod.findglobaltags(self.ui, self)
2024 alltags = tagsmod.findglobaltags(self.ui, self)
2025 # map tag name to tag type
2025 # map tag name to tag type
2026 tagtypes = {tag: b'global' for tag in alltags}
2026 tagtypes = {tag: b'global' for tag in alltags}
2027
2027
2028 tagsmod.readlocaltags(self.ui, self, alltags, tagtypes)
2028 tagsmod.readlocaltags(self.ui, self, alltags, tagtypes)
2029
2029
2030 # Build the return dicts. Have to re-encode tag names because
2030 # Build the return dicts. Have to re-encode tag names because
2031 # the tags module always uses UTF-8 (in order not to lose info
2031 # the tags module always uses UTF-8 (in order not to lose info
2032 # writing to the cache), but the rest of Mercurial wants them in
2032 # writing to the cache), but the rest of Mercurial wants them in
2033 # local encoding.
2033 # local encoding.
2034 tags = {}
2034 tags = {}
2035 for (name, (node, hist)) in pycompat.iteritems(alltags):
2035 for (name, (node, hist)) in pycompat.iteritems(alltags):
2036 if node != nullid:
2036 if node != nullid:
2037 tags[encoding.tolocal(name)] = node
2037 tags[encoding.tolocal(name)] = node
2038 tags[b'tip'] = self.changelog.tip()
2038 tags[b'tip'] = self.changelog.tip()
2039 tagtypes = {
2039 tagtypes = {
2040 encoding.tolocal(name): value
2040 encoding.tolocal(name): value
2041 for (name, value) in pycompat.iteritems(tagtypes)
2041 for (name, value) in pycompat.iteritems(tagtypes)
2042 }
2042 }
2043 return (tags, tagtypes)
2043 return (tags, tagtypes)
2044
2044
2045 def tagtype(self, tagname):
2045 def tagtype(self, tagname):
2046 """
2046 """
2047 return the type of the given tag. result can be:
2047 return the type of the given tag. result can be:
2048
2048
2049 'local' : a local tag
2049 'local' : a local tag
2050 'global' : a global tag
2050 'global' : a global tag
2051 None : tag does not exist
2051 None : tag does not exist
2052 """
2052 """
2053
2053
2054 return self._tagscache.tagtypes.get(tagname)
2054 return self._tagscache.tagtypes.get(tagname)
2055
2055
2056 def tagslist(self):
2056 def tagslist(self):
2057 '''return a list of tags ordered by revision'''
2057 '''return a list of tags ordered by revision'''
2058 if not self._tagscache.tagslist:
2058 if not self._tagscache.tagslist:
2059 l = []
2059 l = []
2060 for t, n in pycompat.iteritems(self.tags()):
2060 for t, n in pycompat.iteritems(self.tags()):
2061 l.append((self.changelog.rev(n), t, n))
2061 l.append((self.changelog.rev(n), t, n))
2062 self._tagscache.tagslist = [(t, n) for r, t, n in sorted(l)]
2062 self._tagscache.tagslist = [(t, n) for r, t, n in sorted(l)]
2063
2063
2064 return self._tagscache.tagslist
2064 return self._tagscache.tagslist
2065
2065
2066 def nodetags(self, node):
2066 def nodetags(self, node):
2067 '''return the tags associated with a node'''
2067 '''return the tags associated with a node'''
2068 if not self._tagscache.nodetagscache:
2068 if not self._tagscache.nodetagscache:
2069 nodetagscache = {}
2069 nodetagscache = {}
2070 for t, n in pycompat.iteritems(self._tagscache.tags):
2070 for t, n in pycompat.iteritems(self._tagscache.tags):
2071 nodetagscache.setdefault(n, []).append(t)
2071 nodetagscache.setdefault(n, []).append(t)
2072 for tags in pycompat.itervalues(nodetagscache):
2072 for tags in pycompat.itervalues(nodetagscache):
2073 tags.sort()
2073 tags.sort()
2074 self._tagscache.nodetagscache = nodetagscache
2074 self._tagscache.nodetagscache = nodetagscache
2075 return self._tagscache.nodetagscache.get(node, [])
2075 return self._tagscache.nodetagscache.get(node, [])
2076
2076
2077 def nodebookmarks(self, node):
2077 def nodebookmarks(self, node):
2078 """return the list of bookmarks pointing to the specified node"""
2078 """return the list of bookmarks pointing to the specified node"""
2079 return self._bookmarks.names(node)
2079 return self._bookmarks.names(node)
2080
2080
2081 def branchmap(self):
2081 def branchmap(self):
2082 """returns a dictionary {branch: [branchheads]} with branchheads
2082 """returns a dictionary {branch: [branchheads]} with branchheads
2083 ordered by increasing revision number"""
2083 ordered by increasing revision number"""
2084 return self._branchcaches[self]
2084 return self._branchcaches[self]
2085
2085
2086 @unfilteredmethod
2086 @unfilteredmethod
2087 def revbranchcache(self):
2087 def revbranchcache(self):
2088 if not self._revbranchcache:
2088 if not self._revbranchcache:
2089 self._revbranchcache = branchmap.revbranchcache(self.unfiltered())
2089 self._revbranchcache = branchmap.revbranchcache(self.unfiltered())
2090 return self._revbranchcache
2090 return self._revbranchcache
2091
2091
2092 def register_changeset(self, rev, changelogrevision):
2092 def register_changeset(self, rev, changelogrevision):
2093 self.revbranchcache().setdata(rev, changelogrevision)
2093 self.revbranchcache().setdata(rev, changelogrevision)
2094
2094
2095 def branchtip(self, branch, ignoremissing=False):
2095 def branchtip(self, branch, ignoremissing=False):
2096 """return the tip node for a given branch
2096 """return the tip node for a given branch
2097
2097
2098 If ignoremissing is True, then this method will not raise an error.
2098 If ignoremissing is True, then this method will not raise an error.
2099 This is helpful for callers that only expect None for a missing branch
2099 This is helpful for callers that only expect None for a missing branch
2100 (e.g. namespace).
2100 (e.g. namespace).
2101
2101
2102 """
2102 """
2103 try:
2103 try:
2104 return self.branchmap().branchtip(branch)
2104 return self.branchmap().branchtip(branch)
2105 except KeyError:
2105 except KeyError:
2106 if not ignoremissing:
2106 if not ignoremissing:
2107 raise error.RepoLookupError(_(b"unknown branch '%s'") % branch)
2107 raise error.RepoLookupError(_(b"unknown branch '%s'") % branch)
2108 else:
2108 else:
2109 pass
2109 pass
2110
2110
2111 def lookup(self, key):
2111 def lookup(self, key):
2112 node = scmutil.revsymbol(self, key).node()
2112 node = scmutil.revsymbol(self, key).node()
2113 if node is None:
2113 if node is None:
2114 raise error.RepoLookupError(_(b"unknown revision '%s'") % key)
2114 raise error.RepoLookupError(_(b"unknown revision '%s'") % key)
2115 return node
2115 return node
2116
2116
2117 def lookupbranch(self, key):
2117 def lookupbranch(self, key):
2118 if self.branchmap().hasbranch(key):
2118 if self.branchmap().hasbranch(key):
2119 return key
2119 return key
2120
2120
2121 return scmutil.revsymbol(self, key).branch()
2121 return scmutil.revsymbol(self, key).branch()
2122
2122
2123 def known(self, nodes):
2123 def known(self, nodes):
2124 cl = self.changelog
2124 cl = self.changelog
2125 get_rev = cl.index.get_rev
2125 get_rev = cl.index.get_rev
2126 filtered = cl.filteredrevs
2126 filtered = cl.filteredrevs
2127 result = []
2127 result = []
2128 for n in nodes:
2128 for n in nodes:
2129 r = get_rev(n)
2129 r = get_rev(n)
2130 resp = not (r is None or r in filtered)
2130 resp = not (r is None or r in filtered)
2131 result.append(resp)
2131 result.append(resp)
2132 return result
2132 return result
2133
2133
2134 def local(self):
2134 def local(self):
2135 return self
2135 return self
2136
2136
2137 def publishing(self):
2137 def publishing(self):
2138 # it's safe (and desirable) to trust the publish flag unconditionally
2138 # it's safe (and desirable) to trust the publish flag unconditionally
2139 # so that we don't finalize changes shared between users via ssh or nfs
2139 # so that we don't finalize changes shared between users via ssh or nfs
2140 return self.ui.configbool(b'phases', b'publish', untrusted=True)
2140 return self.ui.configbool(b'phases', b'publish', untrusted=True)
2141
2141
2142 def cancopy(self):
2142 def cancopy(self):
2143 # so statichttprepo's override of local() works
2143 # so statichttprepo's override of local() works
2144 if not self.local():
2144 if not self.local():
2145 return False
2145 return False
2146 if not self.publishing():
2146 if not self.publishing():
2147 return True
2147 return True
2148 # if publishing we can't copy if there is filtered content
2148 # if publishing we can't copy if there is filtered content
2149 return not self.filtered(b'visible').changelog.filteredrevs
2149 return not self.filtered(b'visible').changelog.filteredrevs
2150
2150
2151 def shared(self):
2151 def shared(self):
2152 '''the type of shared repository (None if not shared)'''
2152 '''the type of shared repository (None if not shared)'''
2153 if self.sharedpath != self.path:
2153 if self.sharedpath != self.path:
2154 return b'store'
2154 return b'store'
2155 return None
2155 return None
2156
2156
2157 def wjoin(self, f, *insidef):
2157 def wjoin(self, f, *insidef):
2158 return self.vfs.reljoin(self.root, f, *insidef)
2158 return self.vfs.reljoin(self.root, f, *insidef)
2159
2159
2160 def setparents(self, p1, p2=nullid):
2160 def setparents(self, p1, p2=nullid):
2161 self[None].setparents(p1, p2)
2161 self[None].setparents(p1, p2)
2162 self._quick_access_changeid_invalidate()
2162 self._quick_access_changeid_invalidate()
2163
2163
2164 def filectx(self, path, changeid=None, fileid=None, changectx=None):
2164 def filectx(self, path, changeid=None, fileid=None, changectx=None):
2165 """changeid must be a changeset revision, if specified.
2165 """changeid must be a changeset revision, if specified.
2166 fileid can be a file revision or node."""
2166 fileid can be a file revision or node."""
2167 return context.filectx(
2167 return context.filectx(
2168 self, path, changeid, fileid, changectx=changectx
2168 self, path, changeid, fileid, changectx=changectx
2169 )
2169 )
2170
2170
2171 def getcwd(self):
2171 def getcwd(self):
2172 return self.dirstate.getcwd()
2172 return self.dirstate.getcwd()
2173
2173
2174 def pathto(self, f, cwd=None):
2174 def pathto(self, f, cwd=None):
2175 return self.dirstate.pathto(f, cwd)
2175 return self.dirstate.pathto(f, cwd)
2176
2176
2177 def _loadfilter(self, filter):
2177 def _loadfilter(self, filter):
2178 if filter not in self._filterpats:
2178 if filter not in self._filterpats:
2179 l = []
2179 l = []
2180 for pat, cmd in self.ui.configitems(filter):
2180 for pat, cmd in self.ui.configitems(filter):
2181 if cmd == b'!':
2181 if cmd == b'!':
2182 continue
2182 continue
2183 mf = matchmod.match(self.root, b'', [pat])
2183 mf = matchmod.match(self.root, b'', [pat])
2184 fn = None
2184 fn = None
2185 params = cmd
2185 params = cmd
2186 for name, filterfn in pycompat.iteritems(self._datafilters):
2186 for name, filterfn in pycompat.iteritems(self._datafilters):
2187 if cmd.startswith(name):
2187 if cmd.startswith(name):
2188 fn = filterfn
2188 fn = filterfn
2189 params = cmd[len(name) :].lstrip()
2189 params = cmd[len(name) :].lstrip()
2190 break
2190 break
2191 if not fn:
2191 if not fn:
2192 fn = lambda s, c, **kwargs: procutil.filter(s, c)
2192 fn = lambda s, c, **kwargs: procutil.filter(s, c)
2193 fn.__name__ = 'commandfilter'
2193 fn.__name__ = 'commandfilter'
2194 # Wrap old filters not supporting keyword arguments
2194 # Wrap old filters not supporting keyword arguments
2195 if not pycompat.getargspec(fn)[2]:
2195 if not pycompat.getargspec(fn)[2]:
2196 oldfn = fn
2196 oldfn = fn
2197 fn = lambda s, c, oldfn=oldfn, **kwargs: oldfn(s, c)
2197 fn = lambda s, c, oldfn=oldfn, **kwargs: oldfn(s, c)
2198 fn.__name__ = 'compat-' + oldfn.__name__
2198 fn.__name__ = 'compat-' + oldfn.__name__
2199 l.append((mf, fn, params))
2199 l.append((mf, fn, params))
2200 self._filterpats[filter] = l
2200 self._filterpats[filter] = l
2201 return self._filterpats[filter]
2201 return self._filterpats[filter]
2202
2202
2203 def _filter(self, filterpats, filename, data):
2203 def _filter(self, filterpats, filename, data):
2204 for mf, fn, cmd in filterpats:
2204 for mf, fn, cmd in filterpats:
2205 if mf(filename):
2205 if mf(filename):
2206 self.ui.debug(
2206 self.ui.debug(
2207 b"filtering %s through %s\n"
2207 b"filtering %s through %s\n"
2208 % (filename, cmd or pycompat.sysbytes(fn.__name__))
2208 % (filename, cmd or pycompat.sysbytes(fn.__name__))
2209 )
2209 )
2210 data = fn(data, cmd, ui=self.ui, repo=self, filename=filename)
2210 data = fn(data, cmd, ui=self.ui, repo=self, filename=filename)
2211 break
2211 break
2212
2212
2213 return data
2213 return data
2214
2214
2215 @unfilteredpropertycache
2215 @unfilteredpropertycache
2216 def _encodefilterpats(self):
2216 def _encodefilterpats(self):
2217 return self._loadfilter(b'encode')
2217 return self._loadfilter(b'encode')
2218
2218
2219 @unfilteredpropertycache
2219 @unfilteredpropertycache
2220 def _decodefilterpats(self):
2220 def _decodefilterpats(self):
2221 return self._loadfilter(b'decode')
2221 return self._loadfilter(b'decode')
2222
2222
2223 def adddatafilter(self, name, filter):
2223 def adddatafilter(self, name, filter):
2224 self._datafilters[name] = filter
2224 self._datafilters[name] = filter
2225
2225
2226 def wread(self, filename):
2226 def wread(self, filename):
2227 if self.wvfs.islink(filename):
2227 if self.wvfs.islink(filename):
2228 data = self.wvfs.readlink(filename)
2228 data = self.wvfs.readlink(filename)
2229 else:
2229 else:
2230 data = self.wvfs.read(filename)
2230 data = self.wvfs.read(filename)
2231 return self._filter(self._encodefilterpats, filename, data)
2231 return self._filter(self._encodefilterpats, filename, data)
2232
2232
2233 def wwrite(self, filename, data, flags, backgroundclose=False, **kwargs):
2233 def wwrite(self, filename, data, flags, backgroundclose=False, **kwargs):
2234 """write ``data`` into ``filename`` in the working directory
2234 """write ``data`` into ``filename`` in the working directory
2235
2235
2236 This returns length of written (maybe decoded) data.
2236 This returns length of written (maybe decoded) data.
2237 """
2237 """
2238 data = self._filter(self._decodefilterpats, filename, data)
2238 data = self._filter(self._decodefilterpats, filename, data)
2239 if b'l' in flags:
2239 if b'l' in flags:
2240 self.wvfs.symlink(data, filename)
2240 self.wvfs.symlink(data, filename)
2241 else:
2241 else:
2242 self.wvfs.write(
2242 self.wvfs.write(
2243 filename, data, backgroundclose=backgroundclose, **kwargs
2243 filename, data, backgroundclose=backgroundclose, **kwargs
2244 )
2244 )
2245 if b'x' in flags:
2245 if b'x' in flags:
2246 self.wvfs.setflags(filename, False, True)
2246 self.wvfs.setflags(filename, False, True)
2247 else:
2247 else:
2248 self.wvfs.setflags(filename, False, False)
2248 self.wvfs.setflags(filename, False, False)
2249 return len(data)
2249 return len(data)
2250
2250
2251 def wwritedata(self, filename, data):
2251 def wwritedata(self, filename, data):
2252 return self._filter(self._decodefilterpats, filename, data)
2252 return self._filter(self._decodefilterpats, filename, data)
2253
2253
2254 def currenttransaction(self):
2254 def currenttransaction(self):
2255 """return the current transaction or None if non exists"""
2255 """return the current transaction or None if non exists"""
2256 if self._transref:
2256 if self._transref:
2257 tr = self._transref()
2257 tr = self._transref()
2258 else:
2258 else:
2259 tr = None
2259 tr = None
2260
2260
2261 if tr and tr.running():
2261 if tr and tr.running():
2262 return tr
2262 return tr
2263 return None
2263 return None
2264
2264
2265 def transaction(self, desc, report=None):
2265 def transaction(self, desc, report=None):
2266 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
2266 if self.ui.configbool(b'devel', b'all-warnings') or self.ui.configbool(
2267 b'devel', b'check-locks'
2267 b'devel', b'check-locks'
2268 ):
2268 ):
2269 if self._currentlock(self._lockref) is None:
2269 if self._currentlock(self._lockref) is None:
2270 raise error.ProgrammingError(b'transaction requires locking')
2270 raise error.ProgrammingError(b'transaction requires locking')
2271 tr = self.currenttransaction()
2271 tr = self.currenttransaction()
2272 if tr is not None:
2272 if tr is not None:
2273 return tr.nest(name=desc)
2273 return tr.nest(name=desc)
2274
2274
2275 # abort here if the journal already exists
2275 # abort here if the journal already exists
2276 if self.svfs.exists(b"journal"):
2276 if self.svfs.exists(b"journal"):
2277 raise error.RepoError(
2277 raise error.RepoError(
2278 _(b"abandoned transaction found"),
2278 _(b"abandoned transaction found"),
2279 hint=_(b"run 'hg recover' to clean up transaction"),
2279 hint=_(b"run 'hg recover' to clean up transaction"),
2280 )
2280 )
2281
2281
2282 idbase = b"%.40f#%f" % (random.random(), time.time())
2282 idbase = b"%.40f#%f" % (random.random(), time.time())
2283 ha = hex(hashutil.sha1(idbase).digest())
2283 ha = hex(hashutil.sha1(idbase).digest())
2284 txnid = b'TXN:' + ha
2284 txnid = b'TXN:' + ha
2285 self.hook(b'pretxnopen', throw=True, txnname=desc, txnid=txnid)
2285 self.hook(b'pretxnopen', throw=True, txnname=desc, txnid=txnid)
2286
2286
2287 self._writejournal(desc)
2287 self._writejournal(desc)
2288 renames = [(vfs, x, undoname(x)) for vfs, x in self._journalfiles()]
2288 renames = [(vfs, x, undoname(x)) for vfs, x in self._journalfiles()]
2289 if report:
2289 if report:
2290 rp = report
2290 rp = report
2291 else:
2291 else:
2292 rp = self.ui.warn
2292 rp = self.ui.warn
2293 vfsmap = {b'plain': self.vfs, b'store': self.svfs} # root of .hg/
2293 vfsmap = {b'plain': self.vfs, b'store': self.svfs} # root of .hg/
2294 # we must avoid cyclic reference between repo and transaction.
2294 # we must avoid cyclic reference between repo and transaction.
2295 reporef = weakref.ref(self)
2295 reporef = weakref.ref(self)
2296 # Code to track tag movement
2296 # Code to track tag movement
2297 #
2297 #
2298 # Since tags are all handled as file content, it is actually quite hard
2298 # Since tags are all handled as file content, it is actually quite hard
2299 # to track these movement from a code perspective. So we fallback to a
2299 # to track these movement from a code perspective. So we fallback to a
2300 # tracking at the repository level. One could envision to track changes
2300 # tracking at the repository level. One could envision to track changes
2301 # to the '.hgtags' file through changegroup apply but that fails to
2301 # to the '.hgtags' file through changegroup apply but that fails to
2302 # cope with case where transaction expose new heads without changegroup
2302 # cope with case where transaction expose new heads without changegroup
2303 # being involved (eg: phase movement).
2303 # being involved (eg: phase movement).
2304 #
2304 #
2305 # For now, We gate the feature behind a flag since this likely comes
2305 # For now, We gate the feature behind a flag since this likely comes
2306 # with performance impacts. The current code run more often than needed
2306 # with performance impacts. The current code run more often than needed
2307 # and do not use caches as much as it could. The current focus is on
2307 # and do not use caches as much as it could. The current focus is on
2308 # the behavior of the feature so we disable it by default. The flag
2308 # the behavior of the feature so we disable it by default. The flag
2309 # will be removed when we are happy with the performance impact.
2309 # will be removed when we are happy with the performance impact.
2310 #
2310 #
2311 # Once this feature is no longer experimental move the following
2311 # Once this feature is no longer experimental move the following
2312 # documentation to the appropriate help section:
2312 # documentation to the appropriate help section:
2313 #
2313 #
2314 # The ``HG_TAG_MOVED`` variable will be set if the transaction touched
2314 # The ``HG_TAG_MOVED`` variable will be set if the transaction touched
2315 # tags (new or changed or deleted tags). In addition the details of
2315 # tags (new or changed or deleted tags). In addition the details of
2316 # these changes are made available in a file at:
2316 # these changes are made available in a file at:
2317 # ``REPOROOT/.hg/changes/tags.changes``.
2317 # ``REPOROOT/.hg/changes/tags.changes``.
2318 # Make sure you check for HG_TAG_MOVED before reading that file as it
2318 # Make sure you check for HG_TAG_MOVED before reading that file as it
2319 # might exist from a previous transaction even if no tag were touched
2319 # might exist from a previous transaction even if no tag were touched
2320 # in this one. Changes are recorded in a line base format::
2320 # in this one. Changes are recorded in a line base format::
2321 #
2321 #
2322 # <action> <hex-node> <tag-name>\n
2322 # <action> <hex-node> <tag-name>\n
2323 #
2323 #
2324 # Actions are defined as follow:
2324 # Actions are defined as follow:
2325 # "-R": tag is removed,
2325 # "-R": tag is removed,
2326 # "+A": tag is added,
2326 # "+A": tag is added,
2327 # "-M": tag is moved (old value),
2327 # "-M": tag is moved (old value),
2328 # "+M": tag is moved (new value),
2328 # "+M": tag is moved (new value),
2329 tracktags = lambda x: None
2329 tracktags = lambda x: None
2330 # experimental config: experimental.hook-track-tags
2330 # experimental config: experimental.hook-track-tags
2331 shouldtracktags = self.ui.configbool(
2331 shouldtracktags = self.ui.configbool(
2332 b'experimental', b'hook-track-tags'
2332 b'experimental', b'hook-track-tags'
2333 )
2333 )
2334 if desc != b'strip' and shouldtracktags:
2334 if desc != b'strip' and shouldtracktags:
2335 oldheads = self.changelog.headrevs()
2335 oldheads = self.changelog.headrevs()
2336
2336
2337 def tracktags(tr2):
2337 def tracktags(tr2):
2338 repo = reporef()
2338 repo = reporef()
2339 assert repo is not None # help pytype
2339 assert repo is not None # help pytype
2340 oldfnodes = tagsmod.fnoderevs(repo.ui, repo, oldheads)
2340 oldfnodes = tagsmod.fnoderevs(repo.ui, repo, oldheads)
2341 newheads = repo.changelog.headrevs()
2341 newheads = repo.changelog.headrevs()
2342 newfnodes = tagsmod.fnoderevs(repo.ui, repo, newheads)
2342 newfnodes = tagsmod.fnoderevs(repo.ui, repo, newheads)
2343 # notes: we compare lists here.
2343 # notes: we compare lists here.
2344 # As we do it only once buiding set would not be cheaper
2344 # As we do it only once buiding set would not be cheaper
2345 changes = tagsmod.difftags(repo.ui, repo, oldfnodes, newfnodes)
2345 changes = tagsmod.difftags(repo.ui, repo, oldfnodes, newfnodes)
2346 if changes:
2346 if changes:
2347 tr2.hookargs[b'tag_moved'] = b'1'
2347 tr2.hookargs[b'tag_moved'] = b'1'
2348 with repo.vfs(
2348 with repo.vfs(
2349 b'changes/tags.changes', b'w', atomictemp=True
2349 b'changes/tags.changes', b'w', atomictemp=True
2350 ) as changesfile:
2350 ) as changesfile:
2351 # note: we do not register the file to the transaction
2351 # note: we do not register the file to the transaction
2352 # because we needs it to still exist on the transaction
2352 # because we needs it to still exist on the transaction
2353 # is close (for txnclose hooks)
2353 # is close (for txnclose hooks)
2354 tagsmod.writediff(changesfile, changes)
2354 tagsmod.writediff(changesfile, changes)
2355
2355
2356 def validate(tr2):
2356 def validate(tr2):
2357 """will run pre-closing hooks"""
2357 """will run pre-closing hooks"""
2358 # XXX the transaction API is a bit lacking here so we take a hacky
2358 # XXX the transaction API is a bit lacking here so we take a hacky
2359 # path for now
2359 # path for now
2360 #
2360 #
2361 # We cannot add this as a "pending" hooks since the 'tr.hookargs'
2361 # We cannot add this as a "pending" hooks since the 'tr.hookargs'
2362 # dict is copied before these run. In addition we needs the data
2362 # dict is copied before these run. In addition we needs the data
2363 # available to in memory hooks too.
2363 # available to in memory hooks too.
2364 #
2364 #
2365 # Moreover, we also need to make sure this runs before txnclose
2365 # Moreover, we also need to make sure this runs before txnclose
2366 # hooks and there is no "pending" mechanism that would execute
2366 # hooks and there is no "pending" mechanism that would execute
2367 # logic only if hooks are about to run.
2367 # logic only if hooks are about to run.
2368 #
2368 #
2369 # Fixing this limitation of the transaction is also needed to track
2369 # Fixing this limitation of the transaction is also needed to track
2370 # other families of changes (bookmarks, phases, obsolescence).
2370 # other families of changes (bookmarks, phases, obsolescence).
2371 #
2371 #
2372 # This will have to be fixed before we remove the experimental
2372 # This will have to be fixed before we remove the experimental
2373 # gating.
2373 # gating.
2374 tracktags(tr2)
2374 tracktags(tr2)
2375 repo = reporef()
2375 repo = reporef()
2376 assert repo is not None # help pytype
2376 assert repo is not None # help pytype
2377
2377
2378 singleheadopt = (b'experimental', b'single-head-per-branch')
2378 singleheadopt = (b'experimental', b'single-head-per-branch')
2379 singlehead = repo.ui.configbool(*singleheadopt)
2379 singlehead = repo.ui.configbool(*singleheadopt)
2380 if singlehead:
2380 if singlehead:
2381 singleheadsub = repo.ui.configsuboptions(*singleheadopt)[1]
2381 singleheadsub = repo.ui.configsuboptions(*singleheadopt)[1]
2382 accountclosed = singleheadsub.get(
2382 accountclosed = singleheadsub.get(
2383 b"account-closed-heads", False
2383 b"account-closed-heads", False
2384 )
2384 )
2385 if singleheadsub.get(b"public-changes-only", False):
2385 if singleheadsub.get(b"public-changes-only", False):
2386 filtername = b"immutable"
2386 filtername = b"immutable"
2387 else:
2387 else:
2388 filtername = b"visible"
2388 filtername = b"visible"
2389 scmutil.enforcesinglehead(
2389 scmutil.enforcesinglehead(
2390 repo, tr2, desc, accountclosed, filtername
2390 repo, tr2, desc, accountclosed, filtername
2391 )
2391 )
2392 if hook.hashook(repo.ui, b'pretxnclose-bookmark'):
2392 if hook.hashook(repo.ui, b'pretxnclose-bookmark'):
2393 for name, (old, new) in sorted(
2393 for name, (old, new) in sorted(
2394 tr.changes[b'bookmarks'].items()
2394 tr.changes[b'bookmarks'].items()
2395 ):
2395 ):
2396 args = tr.hookargs.copy()
2396 args = tr.hookargs.copy()
2397 args.update(bookmarks.preparehookargs(name, old, new))
2397 args.update(bookmarks.preparehookargs(name, old, new))
2398 repo.hook(
2398 repo.hook(
2399 b'pretxnclose-bookmark',
2399 b'pretxnclose-bookmark',
2400 throw=True,
2400 throw=True,
2401 **pycompat.strkwargs(args)
2401 **pycompat.strkwargs(args)
2402 )
2402 )
2403 if hook.hashook(repo.ui, b'pretxnclose-phase'):
2403 if hook.hashook(repo.ui, b'pretxnclose-phase'):
2404 cl = repo.unfiltered().changelog
2404 cl = repo.unfiltered().changelog
2405 for revs, (old, new) in tr.changes[b'phases']:
2405 for revs, (old, new) in tr.changes[b'phases']:
2406 for rev in revs:
2406 for rev in revs:
2407 args = tr.hookargs.copy()
2407 args = tr.hookargs.copy()
2408 node = hex(cl.node(rev))
2408 node = hex(cl.node(rev))
2409 args.update(phases.preparehookargs(node, old, new))
2409 args.update(phases.preparehookargs(node, old, new))
2410 repo.hook(
2410 repo.hook(
2411 b'pretxnclose-phase',
2411 b'pretxnclose-phase',
2412 throw=True,
2412 throw=True,
2413 **pycompat.strkwargs(args)
2413 **pycompat.strkwargs(args)
2414 )
2414 )
2415
2415
2416 repo.hook(
2416 repo.hook(
2417 b'pretxnclose', throw=True, **pycompat.strkwargs(tr.hookargs)
2417 b'pretxnclose', throw=True, **pycompat.strkwargs(tr.hookargs)
2418 )
2418 )
2419
2419
2420 def releasefn(tr, success):
2420 def releasefn(tr, success):
2421 repo = reporef()
2421 repo = reporef()
2422 if repo is None:
2422 if repo is None:
2423 # If the repo has been GC'd (and this release function is being
2423 # If the repo has been GC'd (and this release function is being
2424 # called from transaction.__del__), there's not much we can do,
2424 # called from transaction.__del__), there's not much we can do,
2425 # so just leave the unfinished transaction there and let the
2425 # so just leave the unfinished transaction there and let the
2426 # user run `hg recover`.
2426 # user run `hg recover`.
2427 return
2427 return
2428 if success:
2428 if success:
2429 # this should be explicitly invoked here, because
2429 # this should be explicitly invoked here, because
2430 # in-memory changes aren't written out at closing
2430 # in-memory changes aren't written out at closing
2431 # transaction, if tr.addfilegenerator (via
2431 # transaction, if tr.addfilegenerator (via
2432 # dirstate.write or so) isn't invoked while
2432 # dirstate.write or so) isn't invoked while
2433 # transaction running
2433 # transaction running
2434 repo.dirstate.write(None)
2434 repo.dirstate.write(None)
2435 else:
2435 else:
2436 # discard all changes (including ones already written
2436 # discard all changes (including ones already written
2437 # out) in this transaction
2437 # out) in this transaction
2438 narrowspec.restorebackup(self, b'journal.narrowspec')
2438 narrowspec.restorebackup(self, b'journal.narrowspec')
2439 narrowspec.restorewcbackup(self, b'journal.narrowspec.dirstate')
2439 narrowspec.restorewcbackup(self, b'journal.narrowspec.dirstate')
2440 repo.dirstate.restorebackup(None, b'journal.dirstate')
2440 repo.dirstate.restorebackup(None, b'journal.dirstate')
2441
2441
2442 repo.invalidate(clearfilecache=True)
2442 repo.invalidate(clearfilecache=True)
2443
2443
2444 tr = transaction.transaction(
2444 tr = transaction.transaction(
2445 rp,
2445 rp,
2446 self.svfs,
2446 self.svfs,
2447 vfsmap,
2447 vfsmap,
2448 b"journal",
2448 b"journal",
2449 b"undo",
2449 b"undo",
2450 aftertrans(renames),
2450 aftertrans(renames),
2451 self.store.createmode,
2451 self.store.createmode,
2452 validator=validate,
2452 validator=validate,
2453 releasefn=releasefn,
2453 releasefn=releasefn,
2454 checkambigfiles=_cachedfiles,
2454 checkambigfiles=_cachedfiles,
2455 name=desc,
2455 name=desc,
2456 )
2456 )
2457 tr.changes[b'origrepolen'] = len(self)
2457 tr.changes[b'origrepolen'] = len(self)
2458 tr.changes[b'obsmarkers'] = set()
2458 tr.changes[b'obsmarkers'] = set()
2459 tr.changes[b'phases'] = []
2459 tr.changes[b'phases'] = []
2460 tr.changes[b'bookmarks'] = {}
2460 tr.changes[b'bookmarks'] = {}
2461
2461
2462 tr.hookargs[b'txnid'] = txnid
2462 tr.hookargs[b'txnid'] = txnid
2463 tr.hookargs[b'txnname'] = desc
2463 tr.hookargs[b'txnname'] = desc
2464 tr.hookargs[b'changes'] = tr.changes
2464 tr.hookargs[b'changes'] = tr.changes
2465 # note: writing the fncache only during finalize mean that the file is
2465 # note: writing the fncache only during finalize mean that the file is
2466 # outdated when running hooks. As fncache is used for streaming clone,
2466 # outdated when running hooks. As fncache is used for streaming clone,
2467 # this is not expected to break anything that happen during the hooks.
2467 # this is not expected to break anything that happen during the hooks.
2468 tr.addfinalize(b'flush-fncache', self.store.write)
2468 tr.addfinalize(b'flush-fncache', self.store.write)
2469
2469
2470 def txnclosehook(tr2):
2470 def txnclosehook(tr2):
2471 """To be run if transaction is successful, will schedule a hook run"""
2471 """To be run if transaction is successful, will schedule a hook run"""
2472 # Don't reference tr2 in hook() so we don't hold a reference.
2472 # Don't reference tr2 in hook() so we don't hold a reference.
2473 # This reduces memory consumption when there are multiple
2473 # This reduces memory consumption when there are multiple
2474 # transactions per lock. This can likely go away if issue5045
2474 # transactions per lock. This can likely go away if issue5045
2475 # fixes the function accumulation.
2475 # fixes the function accumulation.
2476 hookargs = tr2.hookargs
2476 hookargs = tr2.hookargs
2477
2477
2478 def hookfunc(unused_success):
2478 def hookfunc(unused_success):
2479 repo = reporef()
2479 repo = reporef()
2480 assert repo is not None # help pytype
2480 assert repo is not None # help pytype
2481
2481
2482 if hook.hashook(repo.ui, b'txnclose-bookmark'):
2482 if hook.hashook(repo.ui, b'txnclose-bookmark'):
2483 bmchanges = sorted(tr.changes[b'bookmarks'].items())
2483 bmchanges = sorted(tr.changes[b'bookmarks'].items())
2484 for name, (old, new) in bmchanges:
2484 for name, (old, new) in bmchanges:
2485 args = tr.hookargs.copy()
2485 args = tr.hookargs.copy()
2486 args.update(bookmarks.preparehookargs(name, old, new))
2486 args.update(bookmarks.preparehookargs(name, old, new))
2487 repo.hook(
2487 repo.hook(
2488 b'txnclose-bookmark',
2488 b'txnclose-bookmark',
2489 throw=False,
2489 throw=False,
2490 **pycompat.strkwargs(args)
2490 **pycompat.strkwargs(args)
2491 )
2491 )
2492
2492
2493 if hook.hashook(repo.ui, b'txnclose-phase'):
2493 if hook.hashook(repo.ui, b'txnclose-phase'):
2494 cl = repo.unfiltered().changelog
2494 cl = repo.unfiltered().changelog
2495 phasemv = sorted(
2495 phasemv = sorted(
2496 tr.changes[b'phases'], key=lambda r: r[0][0]
2496 tr.changes[b'phases'], key=lambda r: r[0][0]
2497 )
2497 )
2498 for revs, (old, new) in phasemv:
2498 for revs, (old, new) in phasemv:
2499 for rev in revs:
2499 for rev in revs:
2500 args = tr.hookargs.copy()
2500 args = tr.hookargs.copy()
2501 node = hex(cl.node(rev))
2501 node = hex(cl.node(rev))
2502 args.update(phases.preparehookargs(node, old, new))
2502 args.update(phases.preparehookargs(node, old, new))
2503 repo.hook(
2503 repo.hook(
2504 b'txnclose-phase',
2504 b'txnclose-phase',
2505 throw=False,
2505 throw=False,
2506 **pycompat.strkwargs(args)
2506 **pycompat.strkwargs(args)
2507 )
2507 )
2508
2508
2509 repo.hook(
2509 repo.hook(
2510 b'txnclose', throw=False, **pycompat.strkwargs(hookargs)
2510 b'txnclose', throw=False, **pycompat.strkwargs(hookargs)
2511 )
2511 )
2512
2512
2513 repo = reporef()
2513 repo = reporef()
2514 assert repo is not None # help pytype
2514 assert repo is not None # help pytype
2515 repo._afterlock(hookfunc)
2515 repo._afterlock(hookfunc)
2516
2516
2517 tr.addfinalize(b'txnclose-hook', txnclosehook)
2517 tr.addfinalize(b'txnclose-hook', txnclosehook)
2518 # Include a leading "-" to make it happen before the transaction summary
2518 # Include a leading "-" to make it happen before the transaction summary
2519 # reports registered via scmutil.registersummarycallback() whose names
2519 # reports registered via scmutil.registersummarycallback() whose names
2520 # are 00-txnreport etc. That way, the caches will be warm when the
2520 # are 00-txnreport etc. That way, the caches will be warm when the
2521 # callbacks run.
2521 # callbacks run.
2522 tr.addpostclose(b'-warm-cache', self._buildcacheupdater(tr))
2522 tr.addpostclose(b'-warm-cache', self._buildcacheupdater(tr))
2523
2523
2524 def txnaborthook(tr2):
2524 def txnaborthook(tr2):
2525 """To be run if transaction is aborted"""
2525 """To be run if transaction is aborted"""
2526 repo = reporef()
2526 repo = reporef()
2527 assert repo is not None # help pytype
2527 assert repo is not None # help pytype
2528 repo.hook(
2528 repo.hook(
2529 b'txnabort', throw=False, **pycompat.strkwargs(tr2.hookargs)
2529 b'txnabort', throw=False, **pycompat.strkwargs(tr2.hookargs)
2530 )
2530 )
2531
2531
2532 tr.addabort(b'txnabort-hook', txnaborthook)
2532 tr.addabort(b'txnabort-hook', txnaborthook)
2533 # avoid eager cache invalidation. in-memory data should be identical
2533 # avoid eager cache invalidation. in-memory data should be identical
2534 # to stored data if transaction has no error.
2534 # to stored data if transaction has no error.
2535 tr.addpostclose(b'refresh-filecachestats', self._refreshfilecachestats)
2535 tr.addpostclose(b'refresh-filecachestats', self._refreshfilecachestats)
2536 self._transref = weakref.ref(tr)
2536 self._transref = weakref.ref(tr)
2537 scmutil.registersummarycallback(self, tr, desc)
2537 scmutil.registersummarycallback(self, tr, desc)
2538 return tr
2538 return tr
2539
2539
2540 def _journalfiles(self):
2540 def _journalfiles(self):
2541 return (
2541 return (
2542 (self.svfs, b'journal'),
2542 (self.svfs, b'journal'),
2543 (self.svfs, b'journal.narrowspec'),
2543 (self.svfs, b'journal.narrowspec'),
2544 (self.vfs, b'journal.narrowspec.dirstate'),
2544 (self.vfs, b'journal.narrowspec.dirstate'),
2545 (self.vfs, b'journal.dirstate'),
2545 (self.vfs, b'journal.dirstate'),
2546 (self.vfs, b'journal.branch'),
2546 (self.vfs, b'journal.branch'),
2547 (self.vfs, b'journal.desc'),
2547 (self.vfs, b'journal.desc'),
2548 (bookmarks.bookmarksvfs(self), b'journal.bookmarks'),
2548 (bookmarks.bookmarksvfs(self), b'journal.bookmarks'),
2549 (self.svfs, b'journal.phaseroots'),
2549 (self.svfs, b'journal.phaseroots'),
2550 )
2550 )
2551
2551
2552 def undofiles(self):
2552 def undofiles(self):
2553 return [(vfs, undoname(x)) for vfs, x in self._journalfiles()]
2553 return [(vfs, undoname(x)) for vfs, x in self._journalfiles()]
2554
2554
2555 @unfilteredmethod
2555 @unfilteredmethod
2556 def _writejournal(self, desc):
2556 def _writejournal(self, desc):
2557 self.dirstate.savebackup(None, b'journal.dirstate')
2557 self.dirstate.savebackup(None, b'journal.dirstate')
2558 narrowspec.savewcbackup(self, b'journal.narrowspec.dirstate')
2558 narrowspec.savewcbackup(self, b'journal.narrowspec.dirstate')
2559 narrowspec.savebackup(self, b'journal.narrowspec')
2559 narrowspec.savebackup(self, b'journal.narrowspec')
2560 self.vfs.write(
2560 self.vfs.write(
2561 b"journal.branch", encoding.fromlocal(self.dirstate.branch())
2561 b"journal.branch", encoding.fromlocal(self.dirstate.branch())
2562 )
2562 )
2563 self.vfs.write(b"journal.desc", b"%d\n%s\n" % (len(self), desc))
2563 self.vfs.write(b"journal.desc", b"%d\n%s\n" % (len(self), desc))
2564 bookmarksvfs = bookmarks.bookmarksvfs(self)
2564 bookmarksvfs = bookmarks.bookmarksvfs(self)
2565 bookmarksvfs.write(
2565 bookmarksvfs.write(
2566 b"journal.bookmarks", bookmarksvfs.tryread(b"bookmarks")
2566 b"journal.bookmarks", bookmarksvfs.tryread(b"bookmarks")
2567 )
2567 )
2568 self.svfs.write(b"journal.phaseroots", self.svfs.tryread(b"phaseroots"))
2568 self.svfs.write(b"journal.phaseroots", self.svfs.tryread(b"phaseroots"))
2569
2569
2570 def recover(self):
2570 def recover(self):
2571 with self.lock():
2571 with self.lock():
2572 if self.svfs.exists(b"journal"):
2572 if self.svfs.exists(b"journal"):
2573 self.ui.status(_(b"rolling back interrupted transaction\n"))
2573 self.ui.status(_(b"rolling back interrupted transaction\n"))
2574 vfsmap = {
2574 vfsmap = {
2575 b'': self.svfs,
2575 b'': self.svfs,
2576 b'plain': self.vfs,
2576 b'plain': self.vfs,
2577 }
2577 }
2578 transaction.rollback(
2578 transaction.rollback(
2579 self.svfs,
2579 self.svfs,
2580 vfsmap,
2580 vfsmap,
2581 b"journal",
2581 b"journal",
2582 self.ui.warn,
2582 self.ui.warn,
2583 checkambigfiles=_cachedfiles,
2583 checkambigfiles=_cachedfiles,
2584 )
2584 )
2585 self.invalidate()
2585 self.invalidate()
2586 return True
2586 return True
2587 else:
2587 else:
2588 self.ui.warn(_(b"no interrupted transaction available\n"))
2588 self.ui.warn(_(b"no interrupted transaction available\n"))
2589 return False
2589 return False
2590
2590
2591 def rollback(self, dryrun=False, force=False):
2591 def rollback(self, dryrun=False, force=False):
2592 wlock = lock = dsguard = None
2592 wlock = lock = dsguard = None
2593 try:
2593 try:
2594 wlock = self.wlock()
2594 wlock = self.wlock()
2595 lock = self.lock()
2595 lock = self.lock()
2596 if self.svfs.exists(b"undo"):
2596 if self.svfs.exists(b"undo"):
2597 dsguard = dirstateguard.dirstateguard(self, b'rollback')
2597 dsguard = dirstateguard.dirstateguard(self, b'rollback')
2598
2598
2599 return self._rollback(dryrun, force, dsguard)
2599 return self._rollback(dryrun, force, dsguard)
2600 else:
2600 else:
2601 self.ui.warn(_(b"no rollback information available\n"))
2601 self.ui.warn(_(b"no rollback information available\n"))
2602 return 1
2602 return 1
2603 finally:
2603 finally:
2604 release(dsguard, lock, wlock)
2604 release(dsguard, lock, wlock)
2605
2605
2606 @unfilteredmethod # Until we get smarter cache management
2606 @unfilteredmethod # Until we get smarter cache management
2607 def _rollback(self, dryrun, force, dsguard):
2607 def _rollback(self, dryrun, force, dsguard):
2608 ui = self.ui
2608 ui = self.ui
2609 try:
2609 try:
2610 args = self.vfs.read(b'undo.desc').splitlines()
2610 args = self.vfs.read(b'undo.desc').splitlines()
2611 (oldlen, desc, detail) = (int(args[0]), args[1], None)
2611 (oldlen, desc, detail) = (int(args[0]), args[1], None)
2612 if len(args) >= 3:
2612 if len(args) >= 3:
2613 detail = args[2]
2613 detail = args[2]
2614 oldtip = oldlen - 1
2614 oldtip = oldlen - 1
2615
2615
2616 if detail and ui.verbose:
2616 if detail and ui.verbose:
2617 msg = _(
2617 msg = _(
2618 b'repository tip rolled back to revision %d'
2618 b'repository tip rolled back to revision %d'
2619 b' (undo %s: %s)\n'
2619 b' (undo %s: %s)\n'
2620 ) % (oldtip, desc, detail)
2620 ) % (oldtip, desc, detail)
2621 else:
2621 else:
2622 msg = _(
2622 msg = _(
2623 b'repository tip rolled back to revision %d (undo %s)\n'
2623 b'repository tip rolled back to revision %d (undo %s)\n'
2624 ) % (oldtip, desc)
2624 ) % (oldtip, desc)
2625 except IOError:
2625 except IOError:
2626 msg = _(b'rolling back unknown transaction\n')
2626 msg = _(b'rolling back unknown transaction\n')
2627 desc = None
2627 desc = None
2628
2628
2629 if not force and self[b'.'] != self[b'tip'] and desc == b'commit':
2629 if not force and self[b'.'] != self[b'tip'] and desc == b'commit':
2630 raise error.Abort(
2630 raise error.Abort(
2631 _(
2631 _(
2632 b'rollback of last commit while not checked out '
2632 b'rollback of last commit while not checked out '
2633 b'may lose data'
2633 b'may lose data'
2634 ),
2634 ),
2635 hint=_(b'use -f to force'),
2635 hint=_(b'use -f to force'),
2636 )
2636 )
2637
2637
2638 ui.status(msg)
2638 ui.status(msg)
2639 if dryrun:
2639 if dryrun:
2640 return 0
2640 return 0
2641
2641
2642 parents = self.dirstate.parents()
2642 parents = self.dirstate.parents()
2643 self.destroying()
2643 self.destroying()
2644 vfsmap = {b'plain': self.vfs, b'': self.svfs}
2644 vfsmap = {b'plain': self.vfs, b'': self.svfs}
2645 transaction.rollback(
2645 transaction.rollback(
2646 self.svfs, vfsmap, b'undo', ui.warn, checkambigfiles=_cachedfiles
2646 self.svfs, vfsmap, b'undo', ui.warn, checkambigfiles=_cachedfiles
2647 )
2647 )
2648 bookmarksvfs = bookmarks.bookmarksvfs(self)
2648 bookmarksvfs = bookmarks.bookmarksvfs(self)
2649 if bookmarksvfs.exists(b'undo.bookmarks'):
2649 if bookmarksvfs.exists(b'undo.bookmarks'):
2650 bookmarksvfs.rename(
2650 bookmarksvfs.rename(
2651 b'undo.bookmarks', b'bookmarks', checkambig=True
2651 b'undo.bookmarks', b'bookmarks', checkambig=True
2652 )
2652 )
2653 if self.svfs.exists(b'undo.phaseroots'):
2653 if self.svfs.exists(b'undo.phaseroots'):
2654 self.svfs.rename(b'undo.phaseroots', b'phaseroots', checkambig=True)
2654 self.svfs.rename(b'undo.phaseroots', b'phaseroots', checkambig=True)
2655 self.invalidate()
2655 self.invalidate()
2656
2656
2657 has_node = self.changelog.index.has_node
2657 has_node = self.changelog.index.has_node
2658 parentgone = any(not has_node(p) for p in parents)
2658 parentgone = any(not has_node(p) for p in parents)
2659 if parentgone:
2659 if parentgone:
2660 # prevent dirstateguard from overwriting already restored one
2660 # prevent dirstateguard from overwriting already restored one
2661 dsguard.close()
2661 dsguard.close()
2662
2662
2663 narrowspec.restorebackup(self, b'undo.narrowspec')
2663 narrowspec.restorebackup(self, b'undo.narrowspec')
2664 narrowspec.restorewcbackup(self, b'undo.narrowspec.dirstate')
2664 narrowspec.restorewcbackup(self, b'undo.narrowspec.dirstate')
2665 self.dirstate.restorebackup(None, b'undo.dirstate')
2665 self.dirstate.restorebackup(None, b'undo.dirstate')
2666 try:
2666 try:
2667 branch = self.vfs.read(b'undo.branch')
2667 branch = self.vfs.read(b'undo.branch')
2668 self.dirstate.setbranch(encoding.tolocal(branch))
2668 self.dirstate.setbranch(encoding.tolocal(branch))
2669 except IOError:
2669 except IOError:
2670 ui.warn(
2670 ui.warn(
2671 _(
2671 _(
2672 b'named branch could not be reset: '
2672 b'named branch could not be reset: '
2673 b'current branch is still \'%s\'\n'
2673 b'current branch is still \'%s\'\n'
2674 )
2674 )
2675 % self.dirstate.branch()
2675 % self.dirstate.branch()
2676 )
2676 )
2677
2677
2678 parents = tuple([p.rev() for p in self[None].parents()])
2678 parents = tuple([p.rev() for p in self[None].parents()])
2679 if len(parents) > 1:
2679 if len(parents) > 1:
2680 ui.status(
2680 ui.status(
2681 _(
2681 _(
2682 b'working directory now based on '
2682 b'working directory now based on '
2683 b'revisions %d and %d\n'
2683 b'revisions %d and %d\n'
2684 )
2684 )
2685 % parents
2685 % parents
2686 )
2686 )
2687 else:
2687 else:
2688 ui.status(
2688 ui.status(
2689 _(b'working directory now based on revision %d\n') % parents
2689 _(b'working directory now based on revision %d\n') % parents
2690 )
2690 )
2691 mergestatemod.mergestate.clean(self)
2691 mergestatemod.mergestate.clean(self)
2692
2692
2693 # TODO: if we know which new heads may result from this rollback, pass
2693 # TODO: if we know which new heads may result from this rollback, pass
2694 # them to destroy(), which will prevent the branchhead cache from being
2694 # them to destroy(), which will prevent the branchhead cache from being
2695 # invalidated.
2695 # invalidated.
2696 self.destroyed()
2696 self.destroyed()
2697 return 0
2697 return 0
2698
2698
2699 def _buildcacheupdater(self, newtransaction):
2699 def _buildcacheupdater(self, newtransaction):
2700 """called during transaction to build the callback updating cache
2700 """called during transaction to build the callback updating cache
2701
2701
2702 Lives on the repository to help extension who might want to augment
2702 Lives on the repository to help extension who might want to augment
2703 this logic. For this purpose, the created transaction is passed to the
2703 this logic. For this purpose, the created transaction is passed to the
2704 method.
2704 method.
2705 """
2705 """
2706 # we must avoid cyclic reference between repo and transaction.
2706 # we must avoid cyclic reference between repo and transaction.
2707 reporef = weakref.ref(self)
2707 reporef = weakref.ref(self)
2708
2708
2709 def updater(tr):
2709 def updater(tr):
2710 repo = reporef()
2710 repo = reporef()
2711 assert repo is not None # help pytype
2711 assert repo is not None # help pytype
2712 repo.updatecaches(tr)
2712 repo.updatecaches(tr)
2713
2713
2714 return updater
2714 return updater
2715
2715
2716 @unfilteredmethod
2716 @unfilteredmethod
2717 def updatecaches(self, tr=None, full=False):
2717 def updatecaches(self, tr=None, full=False):
2718 """warm appropriate caches
2718 """warm appropriate caches
2719
2719
2720 If this function is called after a transaction closed. The transaction
2720 If this function is called after a transaction closed. The transaction
2721 will be available in the 'tr' argument. This can be used to selectively
2721 will be available in the 'tr' argument. This can be used to selectively
2722 update caches relevant to the changes in that transaction.
2722 update caches relevant to the changes in that transaction.
2723
2723
2724 If 'full' is set, make sure all caches the function knows about have
2724 If 'full' is set, make sure all caches the function knows about have
2725 up-to-date data. Even the ones usually loaded more lazily.
2725 up-to-date data. Even the ones usually loaded more lazily.
2726 """
2726 """
2727 if tr is not None and tr.hookargs.get(b'source') == b'strip':
2727 if tr is not None and tr.hookargs.get(b'source') == b'strip':
2728 # During strip, many caches are invalid but
2728 # During strip, many caches are invalid but
2729 # later call to `destroyed` will refresh them.
2729 # later call to `destroyed` will refresh them.
2730 return
2730 return
2731
2731
2732 if tr is None or tr.changes[b'origrepolen'] < len(self):
2732 if tr is None or tr.changes[b'origrepolen'] < len(self):
2733 # accessing the 'served' branchmap should refresh all the others,
2733 # accessing the 'served' branchmap should refresh all the others,
2734 self.ui.debug(b'updating the branch cache\n')
2734 self.ui.debug(b'updating the branch cache\n')
2735 self.filtered(b'served').branchmap()
2735 self.filtered(b'served').branchmap()
2736 self.filtered(b'served.hidden').branchmap()
2736 self.filtered(b'served.hidden').branchmap()
2737
2737
2738 if full:
2738 if full:
2739 unfi = self.unfiltered()
2739 unfi = self.unfiltered()
2740
2740
2741 self.changelog.update_caches(transaction=tr)
2741 self.changelog.update_caches(transaction=tr)
2742 self.manifestlog.update_caches(transaction=tr)
2742 self.manifestlog.update_caches(transaction=tr)
2743
2743
2744 rbc = unfi.revbranchcache()
2744 rbc = unfi.revbranchcache()
2745 for r in unfi.changelog:
2745 for r in unfi.changelog:
2746 rbc.branchinfo(r)
2746 rbc.branchinfo(r)
2747 rbc.write()
2747 rbc.write()
2748
2748
2749 # ensure the working copy parents are in the manifestfulltextcache
2749 # ensure the working copy parents are in the manifestfulltextcache
2750 for ctx in self[b'.'].parents():
2750 for ctx in self[b'.'].parents():
2751 ctx.manifest() # accessing the manifest is enough
2751 ctx.manifest() # accessing the manifest is enough
2752
2752
2753 # accessing fnode cache warms the cache
2753 # accessing fnode cache warms the cache
2754 tagsmod.fnoderevs(self.ui, unfi, unfi.changelog.revs())
2754 tagsmod.fnoderevs(self.ui, unfi, unfi.changelog.revs())
2755 # accessing tags warm the cache
2755 # accessing tags warm the cache
2756 self.tags()
2756 self.tags()
2757 self.filtered(b'served').tags()
2757 self.filtered(b'served').tags()
2758
2758
2759 # The `full` arg is documented as updating even the lazily-loaded
2759 # The `full` arg is documented as updating even the lazily-loaded
2760 # caches immediately, so we're forcing a write to cause these caches
2760 # caches immediately, so we're forcing a write to cause these caches
2761 # to be warmed up even if they haven't explicitly been requested
2761 # to be warmed up even if they haven't explicitly been requested
2762 # yet (if they've never been used by hg, they won't ever have been
2762 # yet (if they've never been used by hg, they won't ever have been
2763 # written, even if they're a subset of another kind of cache that
2763 # written, even if they're a subset of another kind of cache that
2764 # *has* been used).
2764 # *has* been used).
2765 for filt in repoview.filtertable.keys():
2765 for filt in repoview.filtertable.keys():
2766 filtered = self.filtered(filt)
2766 filtered = self.filtered(filt)
2767 filtered.branchmap().write(filtered)
2767 filtered.branchmap().write(filtered)
2768
2768
2769 def invalidatecaches(self):
2769 def invalidatecaches(self):
2770
2770
2771 if '_tagscache' in vars(self):
2771 if '_tagscache' in vars(self):
2772 # can't use delattr on proxy
2772 # can't use delattr on proxy
2773 del self.__dict__['_tagscache']
2773 del self.__dict__['_tagscache']
2774
2774
2775 self._branchcaches.clear()
2775 self._branchcaches.clear()
2776 self.invalidatevolatilesets()
2776 self.invalidatevolatilesets()
2777 self._sparsesignaturecache.clear()
2777 self._sparsesignaturecache.clear()
2778
2778
2779 def invalidatevolatilesets(self):
2779 def invalidatevolatilesets(self):
2780 self.filteredrevcache.clear()
2780 self.filteredrevcache.clear()
2781 obsolete.clearobscaches(self)
2781 obsolete.clearobscaches(self)
2782 self._quick_access_changeid_invalidate()
2782 self._quick_access_changeid_invalidate()
2783
2783
2784 def invalidatedirstate(self):
2784 def invalidatedirstate(self):
2785 """Invalidates the dirstate, causing the next call to dirstate
2785 """Invalidates the dirstate, causing the next call to dirstate
2786 to check if it was modified since the last time it was read,
2786 to check if it was modified since the last time it was read,
2787 rereading it if it has.
2787 rereading it if it has.
2788
2788
2789 This is different to dirstate.invalidate() that it doesn't always
2789 This is different to dirstate.invalidate() that it doesn't always
2790 rereads the dirstate. Use dirstate.invalidate() if you want to
2790 rereads the dirstate. Use dirstate.invalidate() if you want to
2791 explicitly read the dirstate again (i.e. restoring it to a previous
2791 explicitly read the dirstate again (i.e. restoring it to a previous
2792 known good state)."""
2792 known good state)."""
2793 if hasunfilteredcache(self, 'dirstate'):
2793 if hasunfilteredcache(self, 'dirstate'):
2794 for k in self.dirstate._filecache:
2794 for k in self.dirstate._filecache:
2795 try:
2795 try:
2796 delattr(self.dirstate, k)
2796 delattr(self.dirstate, k)
2797 except AttributeError:
2797 except AttributeError:
2798 pass
2798 pass
2799 delattr(self.unfiltered(), 'dirstate')
2799 delattr(self.unfiltered(), 'dirstate')
2800
2800
2801 def invalidate(self, clearfilecache=False):
2801 def invalidate(self, clearfilecache=False):
2802 """Invalidates both store and non-store parts other than dirstate
2802 """Invalidates both store and non-store parts other than dirstate
2803
2803
2804 If a transaction is running, invalidation of store is omitted,
2804 If a transaction is running, invalidation of store is omitted,
2805 because discarding in-memory changes might cause inconsistency
2805 because discarding in-memory changes might cause inconsistency
2806 (e.g. incomplete fncache causes unintentional failure, but
2806 (e.g. incomplete fncache causes unintentional failure, but
2807 redundant one doesn't).
2807 redundant one doesn't).
2808 """
2808 """
2809 unfiltered = self.unfiltered() # all file caches are stored unfiltered
2809 unfiltered = self.unfiltered() # all file caches are stored unfiltered
2810 for k in list(self._filecache.keys()):
2810 for k in list(self._filecache.keys()):
2811 # dirstate is invalidated separately in invalidatedirstate()
2811 # dirstate is invalidated separately in invalidatedirstate()
2812 if k == b'dirstate':
2812 if k == b'dirstate':
2813 continue
2813 continue
2814 if (
2814 if (
2815 k == b'changelog'
2815 k == b'changelog'
2816 and self.currenttransaction()
2816 and self.currenttransaction()
2817 and self.changelog._delayed
2817 and self.changelog._delayed
2818 ):
2818 ):
2819 # The changelog object may store unwritten revisions. We don't
2819 # The changelog object may store unwritten revisions. We don't
2820 # want to lose them.
2820 # want to lose them.
2821 # TODO: Solve the problem instead of working around it.
2821 # TODO: Solve the problem instead of working around it.
2822 continue
2822 continue
2823
2823
2824 if clearfilecache:
2824 if clearfilecache:
2825 del self._filecache[k]
2825 del self._filecache[k]
2826 try:
2826 try:
2827 delattr(unfiltered, k)
2827 delattr(unfiltered, k)
2828 except AttributeError:
2828 except AttributeError:
2829 pass
2829 pass
2830 self.invalidatecaches()
2830 self.invalidatecaches()
2831 if not self.currenttransaction():
2831 if not self.currenttransaction():
2832 # TODO: Changing contents of store outside transaction
2832 # TODO: Changing contents of store outside transaction
2833 # causes inconsistency. We should make in-memory store
2833 # causes inconsistency. We should make in-memory store
2834 # changes detectable, and abort if changed.
2834 # changes detectable, and abort if changed.
2835 self.store.invalidatecaches()
2835 self.store.invalidatecaches()
2836
2836
2837 def invalidateall(self):
2837 def invalidateall(self):
2838 """Fully invalidates both store and non-store parts, causing the
2838 """Fully invalidates both store and non-store parts, causing the
2839 subsequent operation to reread any outside changes."""
2839 subsequent operation to reread any outside changes."""
2840 # extension should hook this to invalidate its caches
2840 # extension should hook this to invalidate its caches
2841 self.invalidate()
2841 self.invalidate()
2842 self.invalidatedirstate()
2842 self.invalidatedirstate()
2843
2843
2844 @unfilteredmethod
2844 @unfilteredmethod
2845 def _refreshfilecachestats(self, tr):
2845 def _refreshfilecachestats(self, tr):
2846 """Reload stats of cached files so that they are flagged as valid"""
2846 """Reload stats of cached files so that they are flagged as valid"""
2847 for k, ce in self._filecache.items():
2847 for k, ce in self._filecache.items():
2848 k = pycompat.sysstr(k)
2848 k = pycompat.sysstr(k)
2849 if k == 'dirstate' or k not in self.__dict__:
2849 if k == 'dirstate' or k not in self.__dict__:
2850 continue
2850 continue
2851 ce.refresh()
2851 ce.refresh()
2852
2852
2853 def _lock(
2853 def _lock(
2854 self,
2854 self,
2855 vfs,
2855 vfs,
2856 lockname,
2856 lockname,
2857 wait,
2857 wait,
2858 releasefn,
2858 releasefn,
2859 acquirefn,
2859 acquirefn,
2860 desc,
2860 desc,
2861 ):
2861 ):
2862 timeout = 0
2862 timeout = 0
2863 warntimeout = 0
2863 warntimeout = 0
2864 if wait:
2864 if wait:
2865 timeout = self.ui.configint(b"ui", b"timeout")
2865 timeout = self.ui.configint(b"ui", b"timeout")
2866 warntimeout = self.ui.configint(b"ui", b"timeout.warn")
2866 warntimeout = self.ui.configint(b"ui", b"timeout.warn")
2867 # internal config: ui.signal-safe-lock
2867 # internal config: ui.signal-safe-lock
2868 signalsafe = self.ui.configbool(b'ui', b'signal-safe-lock')
2868 signalsafe = self.ui.configbool(b'ui', b'signal-safe-lock')
2869
2869
2870 l = lockmod.trylock(
2870 l = lockmod.trylock(
2871 self.ui,
2871 self.ui,
2872 vfs,
2872 vfs,
2873 lockname,
2873 lockname,
2874 timeout,
2874 timeout,
2875 warntimeout,
2875 warntimeout,
2876 releasefn=releasefn,
2876 releasefn=releasefn,
2877 acquirefn=acquirefn,
2877 acquirefn=acquirefn,
2878 desc=desc,
2878 desc=desc,
2879 signalsafe=signalsafe,
2879 signalsafe=signalsafe,
2880 )
2880 )
2881 return l
2881 return l
2882
2882
2883 def _afterlock(self, callback):
2883 def _afterlock(self, callback):
2884 """add a callback to be run when the repository is fully unlocked
2884 """add a callback to be run when the repository is fully unlocked
2885
2885
2886 The callback will be executed when the outermost lock is released
2886 The callback will be executed when the outermost lock is released
2887 (with wlock being higher level than 'lock')."""
2887 (with wlock being higher level than 'lock')."""
2888 for ref in (self._wlockref, self._lockref):
2888 for ref in (self._wlockref, self._lockref):
2889 l = ref and ref()
2889 l = ref and ref()
2890 if l and l.held:
2890 if l and l.held:
2891 l.postrelease.append(callback)
2891 l.postrelease.append(callback)
2892 break
2892 break
2893 else: # no lock have been found.
2893 else: # no lock have been found.
2894 callback(True)
2894 callback(True)
2895
2895
2896 def lock(self, wait=True):
2896 def lock(self, wait=True):
2897 """Lock the repository store (.hg/store) and return a weak reference
2897 """Lock the repository store (.hg/store) and return a weak reference
2898 to the lock. Use this before modifying the store (e.g. committing or
2898 to the lock. Use this before modifying the store (e.g. committing or
2899 stripping). If you are opening a transaction, get a lock as well.)
2899 stripping). If you are opening a transaction, get a lock as well.)
2900
2900
2901 If both 'lock' and 'wlock' must be acquired, ensure you always acquires
2901 If both 'lock' and 'wlock' must be acquired, ensure you always acquires
2902 'wlock' first to avoid a dead-lock hazard."""
2902 'wlock' first to avoid a dead-lock hazard."""
2903 l = self._currentlock(self._lockref)
2903 l = self._currentlock(self._lockref)
2904 if l is not None:
2904 if l is not None:
2905 l.lock()
2905 l.lock()
2906 return l
2906 return l
2907
2907
2908 l = self._lock(
2908 l = self._lock(
2909 vfs=self.svfs,
2909 vfs=self.svfs,
2910 lockname=b"lock",
2910 lockname=b"lock",
2911 wait=wait,
2911 wait=wait,
2912 releasefn=None,
2912 releasefn=None,
2913 acquirefn=self.invalidate,
2913 acquirefn=self.invalidate,
2914 desc=_(b'repository %s') % self.origroot,
2914 desc=_(b'repository %s') % self.origroot,
2915 )
2915 )
2916 self._lockref = weakref.ref(l)
2916 self._lockref = weakref.ref(l)
2917 return l
2917 return l
2918
2918
2919 def wlock(self, wait=True):
2919 def wlock(self, wait=True):
2920 """Lock the non-store parts of the repository (everything under
2920 """Lock the non-store parts of the repository (everything under
2921 .hg except .hg/store) and return a weak reference to the lock.
2921 .hg except .hg/store) and return a weak reference to the lock.
2922
2922
2923 Use this before modifying files in .hg.
2923 Use this before modifying files in .hg.
2924
2924
2925 If both 'lock' and 'wlock' must be acquired, ensure you always acquires
2925 If both 'lock' and 'wlock' must be acquired, ensure you always acquires
2926 'wlock' first to avoid a dead-lock hazard."""
2926 'wlock' first to avoid a dead-lock hazard."""
2927 l = self._wlockref() if self._wlockref else None
2927 l = self._wlockref() if self._wlockref else None
2928 if l is not None and l.held:
2928 if l is not None and l.held:
2929 l.lock()
2929 l.lock()
2930 return l
2930 return l
2931
2931
2932 # We do not need to check for non-waiting lock acquisition. Such
2932 # We do not need to check for non-waiting lock acquisition. Such
2933 # acquisition would not cause dead-lock as they would just fail.
2933 # acquisition would not cause dead-lock as they would just fail.
2934 if wait and (
2934 if wait and (
2935 self.ui.configbool(b'devel', b'all-warnings')
2935 self.ui.configbool(b'devel', b'all-warnings')
2936 or self.ui.configbool(b'devel', b'check-locks')
2936 or self.ui.configbool(b'devel', b'check-locks')
2937 ):
2937 ):
2938 if self._currentlock(self._lockref) is not None:
2938 if self._currentlock(self._lockref) is not None:
2939 self.ui.develwarn(b'"wlock" acquired after "lock"')
2939 self.ui.develwarn(b'"wlock" acquired after "lock"')
2940
2940
2941 def unlock():
2941 def unlock():
2942 if self.dirstate.pendingparentchange():
2942 if self.dirstate.pendingparentchange():
2943 self.dirstate.invalidate()
2943 self.dirstate.invalidate()
2944 else:
2944 else:
2945 self.dirstate.write(None)
2945 self.dirstate.write(None)
2946
2946
2947 self._filecache[b'dirstate'].refresh()
2947 self._filecache[b'dirstate'].refresh()
2948
2948
2949 l = self._lock(
2949 l = self._lock(
2950 self.vfs,
2950 self.vfs,
2951 b"wlock",
2951 b"wlock",
2952 wait,
2952 wait,
2953 unlock,
2953 unlock,
2954 self.invalidatedirstate,
2954 self.invalidatedirstate,
2955 _(b'working directory of %s') % self.origroot,
2955 _(b'working directory of %s') % self.origroot,
2956 )
2956 )
2957 self._wlockref = weakref.ref(l)
2957 self._wlockref = weakref.ref(l)
2958 return l
2958 return l
2959
2959
2960 def _currentlock(self, lockref):
2960 def _currentlock(self, lockref):
2961 """Returns the lock if it's held, or None if it's not."""
2961 """Returns the lock if it's held, or None if it's not."""
2962 if lockref is None:
2962 if lockref is None:
2963 return None
2963 return None
2964 l = lockref()
2964 l = lockref()
2965 if l is None or not l.held:
2965 if l is None or not l.held:
2966 return None
2966 return None
2967 return l
2967 return l
2968
2968
2969 def currentwlock(self):
2969 def currentwlock(self):
2970 """Returns the wlock if it's held, or None if it's not."""
2970 """Returns the wlock if it's held, or None if it's not."""
2971 return self._currentlock(self._wlockref)
2971 return self._currentlock(self._wlockref)
2972
2972
2973 def checkcommitpatterns(self, wctx, match, status, fail):
2973 def checkcommitpatterns(self, wctx, match, status, fail):
2974 """check for commit arguments that aren't committable"""
2974 """check for commit arguments that aren't committable"""
2975 if match.isexact() or match.prefix():
2975 if match.isexact() or match.prefix():
2976 matched = set(status.modified + status.added + status.removed)
2976 matched = set(status.modified + status.added + status.removed)
2977
2977
2978 for f in match.files():
2978 for f in match.files():
2979 f = self.dirstate.normalize(f)
2979 f = self.dirstate.normalize(f)
2980 if f == b'.' or f in matched or f in wctx.substate:
2980 if f == b'.' or f in matched or f in wctx.substate:
2981 continue
2981 continue
2982 if f in status.deleted:
2982 if f in status.deleted:
2983 fail(f, _(b'file not found!'))
2983 fail(f, _(b'file not found!'))
2984 # Is it a directory that exists or used to exist?
2984 # Is it a directory that exists or used to exist?
2985 if self.wvfs.isdir(f) or wctx.p1().hasdir(f):
2985 if self.wvfs.isdir(f) or wctx.p1().hasdir(f):
2986 d = f + b'/'
2986 d = f + b'/'
2987 for mf in matched:
2987 for mf in matched:
2988 if mf.startswith(d):
2988 if mf.startswith(d):
2989 break
2989 break
2990 else:
2990 else:
2991 fail(f, _(b"no match under directory!"))
2991 fail(f, _(b"no match under directory!"))
2992 elif f not in self.dirstate:
2992 elif f not in self.dirstate:
2993 fail(f, _(b"file not tracked!"))
2993 fail(f, _(b"file not tracked!"))
2994
2994
2995 @unfilteredmethod
2995 @unfilteredmethod
2996 def commit(
2996 def commit(
2997 self,
2997 self,
2998 text=b"",
2998 text=b"",
2999 user=None,
2999 user=None,
3000 date=None,
3000 date=None,
3001 match=None,
3001 match=None,
3002 force=False,
3002 force=False,
3003 editor=None,
3003 editor=None,
3004 extra=None,
3004 extra=None,
3005 ):
3005 ):
3006 """Add a new revision to current repository.
3006 """Add a new revision to current repository.
3007
3007
3008 Revision information is gathered from the working directory,
3008 Revision information is gathered from the working directory,
3009 match can be used to filter the committed files. If editor is
3009 match can be used to filter the committed files. If editor is
3010 supplied, it is called to get a commit message.
3010 supplied, it is called to get a commit message.
3011 """
3011 """
3012 if extra is None:
3012 if extra is None:
3013 extra = {}
3013 extra = {}
3014
3014
3015 def fail(f, msg):
3015 def fail(f, msg):
3016 raise error.InputError(b'%s: %s' % (f, msg))
3016 raise error.InputError(b'%s: %s' % (f, msg))
3017
3017
3018 if not match:
3018 if not match:
3019 match = matchmod.always()
3019 match = matchmod.always()
3020
3020
3021 if not force:
3021 if not force:
3022 match.bad = fail
3022 match.bad = fail
3023
3023
3024 # lock() for recent changelog (see issue4368)
3024 # lock() for recent changelog (see issue4368)
3025 with self.wlock(), self.lock():
3025 with self.wlock(), self.lock():
3026 wctx = self[None]
3026 wctx = self[None]
3027 merge = len(wctx.parents()) > 1
3027 merge = len(wctx.parents()) > 1
3028
3028
3029 if not force and merge and not match.always():
3029 if not force and merge and not match.always():
3030 raise error.Abort(
3030 raise error.Abort(
3031 _(
3031 _(
3032 b'cannot partially commit a merge '
3032 b'cannot partially commit a merge '
3033 b'(do not specify files or patterns)'
3033 b'(do not specify files or patterns)'
3034 )
3034 )
3035 )
3035 )
3036
3036
3037 status = self.status(match=match, clean=force)
3037 status = self.status(match=match, clean=force)
3038 if force:
3038 if force:
3039 status.modified.extend(
3039 status.modified.extend(
3040 status.clean
3040 status.clean
3041 ) # mq may commit clean files
3041 ) # mq may commit clean files
3042
3042
3043 # check subrepos
3043 # check subrepos
3044 subs, commitsubs, newstate = subrepoutil.precommit(
3044 subs, commitsubs, newstate = subrepoutil.precommit(
3045 self.ui, wctx, status, match, force=force
3045 self.ui, wctx, status, match, force=force
3046 )
3046 )
3047
3047
3048 # make sure all explicit patterns are matched
3048 # make sure all explicit patterns are matched
3049 if not force:
3049 if not force:
3050 self.checkcommitpatterns(wctx, match, status, fail)
3050 self.checkcommitpatterns(wctx, match, status, fail)
3051
3051
3052 cctx = context.workingcommitctx(
3052 cctx = context.workingcommitctx(
3053 self, status, text, user, date, extra
3053 self, status, text, user, date, extra
3054 )
3054 )
3055
3055
3056 ms = mergestatemod.mergestate.read(self)
3056 ms = mergestatemod.mergestate.read(self)
3057 mergeutil.checkunresolved(ms)
3057 mergeutil.checkunresolved(ms)
3058
3058
3059 # internal config: ui.allowemptycommit
3059 # internal config: ui.allowemptycommit
3060 if cctx.isempty() and not self.ui.configbool(
3060 if cctx.isempty() and not self.ui.configbool(
3061 b'ui', b'allowemptycommit'
3061 b'ui', b'allowemptycommit'
3062 ):
3062 ):
3063 self.ui.debug(b'nothing to commit, clearing merge state\n')
3063 self.ui.debug(b'nothing to commit, clearing merge state\n')
3064 ms.reset()
3064 ms.reset()
3065 return None
3065 return None
3066
3066
3067 if merge and cctx.deleted():
3067 if merge and cctx.deleted():
3068 raise error.Abort(_(b"cannot commit merge with missing files"))
3068 raise error.Abort(_(b"cannot commit merge with missing files"))
3069
3069
3070 if editor:
3070 if editor:
3071 cctx._text = editor(self, cctx, subs)
3071 cctx._text = editor(self, cctx, subs)
3072 edited = text != cctx._text
3072 edited = text != cctx._text
3073
3073
3074 # Save commit message in case this transaction gets rolled back
3074 # Save commit message in case this transaction gets rolled back
3075 # (e.g. by a pretxncommit hook). Leave the content alone on
3075 # (e.g. by a pretxncommit hook). Leave the content alone on
3076 # the assumption that the user will use the same editor again.
3076 # the assumption that the user will use the same editor again.
3077 msgfn = self.savecommitmessage(cctx._text)
3077 msgfn = self.savecommitmessage(cctx._text)
3078
3078
3079 # commit subs and write new state
3079 # commit subs and write new state
3080 if subs:
3080 if subs:
3081 uipathfn = scmutil.getuipathfn(self)
3081 uipathfn = scmutil.getuipathfn(self)
3082 for s in sorted(commitsubs):
3082 for s in sorted(commitsubs):
3083 sub = wctx.sub(s)
3083 sub = wctx.sub(s)
3084 self.ui.status(
3084 self.ui.status(
3085 _(b'committing subrepository %s\n')
3085 _(b'committing subrepository %s\n')
3086 % uipathfn(subrepoutil.subrelpath(sub))
3086 % uipathfn(subrepoutil.subrelpath(sub))
3087 )
3087 )
3088 sr = sub.commit(cctx._text, user, date)
3088 sr = sub.commit(cctx._text, user, date)
3089 newstate[s] = (newstate[s][0], sr)
3089 newstate[s] = (newstate[s][0], sr)
3090 subrepoutil.writestate(self, newstate)
3090 subrepoutil.writestate(self, newstate)
3091
3091
3092 p1, p2 = self.dirstate.parents()
3092 p1, p2 = self.dirstate.parents()
3093 hookp1, hookp2 = hex(p1), (p2 != nullid and hex(p2) or b'')
3093 hookp1, hookp2 = hex(p1), (p2 != nullid and hex(p2) or b'')
3094 try:
3094 try:
3095 self.hook(
3095 self.hook(
3096 b"precommit", throw=True, parent1=hookp1, parent2=hookp2
3096 b"precommit", throw=True, parent1=hookp1, parent2=hookp2
3097 )
3097 )
3098 with self.transaction(b'commit'):
3098 with self.transaction(b'commit'):
3099 ret = self.commitctx(cctx, True)
3099 ret = self.commitctx(cctx, True)
3100 # update bookmarks, dirstate and mergestate
3100 # update bookmarks, dirstate and mergestate
3101 bookmarks.update(self, [p1, p2], ret)
3101 bookmarks.update(self, [p1, p2], ret)
3102 cctx.markcommitted(ret)
3102 cctx.markcommitted(ret)
3103 ms.reset()
3103 ms.reset()
3104 except: # re-raises
3104 except: # re-raises
3105 if edited:
3105 if edited:
3106 self.ui.write(
3106 self.ui.write(
3107 _(b'note: commit message saved in %s\n') % msgfn
3107 _(b'note: commit message saved in %s\n') % msgfn
3108 )
3108 )
3109 self.ui.write(
3109 self.ui.write(
3110 _(
3110 _(
3111 b"note: use 'hg commit --logfile "
3111 b"note: use 'hg commit --logfile "
3112 b".hg/last-message.txt --edit' to reuse it\n"
3112 b".hg/last-message.txt --edit' to reuse it\n"
3113 )
3113 )
3114 )
3114 )
3115 raise
3115 raise
3116
3116
3117 def commithook(unused_success):
3117 def commithook(unused_success):
3118 # hack for command that use a temporary commit (eg: histedit)
3118 # hack for command that use a temporary commit (eg: histedit)
3119 # temporary commit got stripped before hook release
3119 # temporary commit got stripped before hook release
3120 if self.changelog.hasnode(ret):
3120 if self.changelog.hasnode(ret):
3121 self.hook(
3121 self.hook(
3122 b"commit", node=hex(ret), parent1=hookp1, parent2=hookp2
3122 b"commit", node=hex(ret), parent1=hookp1, parent2=hookp2
3123 )
3123 )
3124
3124
3125 self._afterlock(commithook)
3125 self._afterlock(commithook)
3126 return ret
3126 return ret
3127
3127
3128 @unfilteredmethod
3128 @unfilteredmethod
3129 def commitctx(self, ctx, error=False, origctx=None):
3129 def commitctx(self, ctx, error=False, origctx=None):
3130 return commit.commitctx(self, ctx, error=error, origctx=origctx)
3130 return commit.commitctx(self, ctx, error=error, origctx=origctx)
3131
3131
3132 @unfilteredmethod
3132 @unfilteredmethod
3133 def destroying(self):
3133 def destroying(self):
3134 """Inform the repository that nodes are about to be destroyed.
3134 """Inform the repository that nodes are about to be destroyed.
3135 Intended for use by strip and rollback, so there's a common
3135 Intended for use by strip and rollback, so there's a common
3136 place for anything that has to be done before destroying history.
3136 place for anything that has to be done before destroying history.
3137
3137
3138 This is mostly useful for saving state that is in memory and waiting
3138 This is mostly useful for saving state that is in memory and waiting
3139 to be flushed when the current lock is released. Because a call to
3139 to be flushed when the current lock is released. Because a call to
3140 destroyed is imminent, the repo will be invalidated causing those
3140 destroyed is imminent, the repo will be invalidated causing those
3141 changes to stay in memory (waiting for the next unlock), or vanish
3141 changes to stay in memory (waiting for the next unlock), or vanish
3142 completely.
3142 completely.
3143 """
3143 """
3144 # When using the same lock to commit and strip, the phasecache is left
3144 # When using the same lock to commit and strip, the phasecache is left
3145 # dirty after committing. Then when we strip, the repo is invalidated,
3145 # dirty after committing. Then when we strip, the repo is invalidated,
3146 # causing those changes to disappear.
3146 # causing those changes to disappear.
3147 if '_phasecache' in vars(self):
3147 if '_phasecache' in vars(self):
3148 self._phasecache.write()
3148 self._phasecache.write()
3149
3149
3150 @unfilteredmethod
3150 @unfilteredmethod
3151 def destroyed(self):
3151 def destroyed(self):
3152 """Inform the repository that nodes have been destroyed.
3152 """Inform the repository that nodes have been destroyed.
3153 Intended for use by strip and rollback, so there's a common
3153 Intended for use by strip and rollback, so there's a common
3154 place for anything that has to be done after destroying history.
3154 place for anything that has to be done after destroying history.
3155 """
3155 """
3156 # When one tries to:
3156 # When one tries to:
3157 # 1) destroy nodes thus calling this method (e.g. strip)
3157 # 1) destroy nodes thus calling this method (e.g. strip)
3158 # 2) use phasecache somewhere (e.g. commit)
3158 # 2) use phasecache somewhere (e.g. commit)
3159 #
3159 #
3160 # then 2) will fail because the phasecache contains nodes that were
3160 # then 2) will fail because the phasecache contains nodes that were
3161 # removed. We can either remove phasecache from the filecache,
3161 # removed. We can either remove phasecache from the filecache,
3162 # causing it to reload next time it is accessed, or simply filter
3162 # causing it to reload next time it is accessed, or simply filter
3163 # the removed nodes now and write the updated cache.
3163 # the removed nodes now and write the updated cache.
3164 self._phasecache.filterunknown(self)
3164 self._phasecache.filterunknown(self)
3165 self._phasecache.write()
3165 self._phasecache.write()
3166
3166
3167 # refresh all repository caches
3167 # refresh all repository caches
3168 self.updatecaches()
3168 self.updatecaches()
3169
3169
3170 # Ensure the persistent tag cache is updated. Doing it now
3170 # Ensure the persistent tag cache is updated. Doing it now
3171 # means that the tag cache only has to worry about destroyed
3171 # means that the tag cache only has to worry about destroyed
3172 # heads immediately after a strip/rollback. That in turn
3172 # heads immediately after a strip/rollback. That in turn
3173 # guarantees that "cachetip == currenttip" (comparing both rev
3173 # guarantees that "cachetip == currenttip" (comparing both rev
3174 # and node) always means no nodes have been added or destroyed.
3174 # and node) always means no nodes have been added or destroyed.
3175
3175
3176 # XXX this is suboptimal when qrefresh'ing: we strip the current
3176 # XXX this is suboptimal when qrefresh'ing: we strip the current
3177 # head, refresh the tag cache, then immediately add a new head.
3177 # head, refresh the tag cache, then immediately add a new head.
3178 # But I think doing it this way is necessary for the "instant
3178 # But I think doing it this way is necessary for the "instant
3179 # tag cache retrieval" case to work.
3179 # tag cache retrieval" case to work.
3180 self.invalidate()
3180 self.invalidate()
3181
3181
3182 def status(
3182 def status(
3183 self,
3183 self,
3184 node1=b'.',
3184 node1=b'.',
3185 node2=None,
3185 node2=None,
3186 match=None,
3186 match=None,
3187 ignored=False,
3187 ignored=False,
3188 clean=False,
3188 clean=False,
3189 unknown=False,
3189 unknown=False,
3190 listsubrepos=False,
3190 listsubrepos=False,
3191 ):
3191 ):
3192 '''a convenience method that calls node1.status(node2)'''
3192 '''a convenience method that calls node1.status(node2)'''
3193 return self[node1].status(
3193 return self[node1].status(
3194 node2, match, ignored, clean, unknown, listsubrepos
3194 node2, match, ignored, clean, unknown, listsubrepos
3195 )
3195 )
3196
3196
3197 def addpostdsstatus(self, ps):
3197 def addpostdsstatus(self, ps):
3198 """Add a callback to run within the wlock, at the point at which status
3198 """Add a callback to run within the wlock, at the point at which status
3199 fixups happen.
3199 fixups happen.
3200
3200
3201 On status completion, callback(wctx, status) will be called with the
3201 On status completion, callback(wctx, status) will be called with the
3202 wlock held, unless the dirstate has changed from underneath or the wlock
3202 wlock held, unless the dirstate has changed from underneath or the wlock
3203 couldn't be grabbed.
3203 couldn't be grabbed.
3204
3204
3205 Callbacks should not capture and use a cached copy of the dirstate --
3205 Callbacks should not capture and use a cached copy of the dirstate --
3206 it might change in the meanwhile. Instead, they should access the
3206 it might change in the meanwhile. Instead, they should access the
3207 dirstate via wctx.repo().dirstate.
3207 dirstate via wctx.repo().dirstate.
3208
3208
3209 This list is emptied out after each status run -- extensions should
3209 This list is emptied out after each status run -- extensions should
3210 make sure it adds to this list each time dirstate.status is called.
3210 make sure it adds to this list each time dirstate.status is called.
3211 Extensions should also make sure they don't call this for statuses
3211 Extensions should also make sure they don't call this for statuses
3212 that don't involve the dirstate.
3212 that don't involve the dirstate.
3213 """
3213 """
3214
3214
3215 # The list is located here for uniqueness reasons -- it is actually
3215 # The list is located here for uniqueness reasons -- it is actually
3216 # managed by the workingctx, but that isn't unique per-repo.
3216 # managed by the workingctx, but that isn't unique per-repo.
3217 self._postdsstatus.append(ps)
3217 self._postdsstatus.append(ps)
3218
3218
3219 def postdsstatus(self):
3219 def postdsstatus(self):
3220 """Used by workingctx to get the list of post-dirstate-status hooks."""
3220 """Used by workingctx to get the list of post-dirstate-status hooks."""
3221 return self._postdsstatus
3221 return self._postdsstatus
3222
3222
3223 def clearpostdsstatus(self):
3223 def clearpostdsstatus(self):
3224 """Used by workingctx to clear post-dirstate-status hooks."""
3224 """Used by workingctx to clear post-dirstate-status hooks."""
3225 del self._postdsstatus[:]
3225 del self._postdsstatus[:]
3226
3226
3227 def heads(self, start=None):
3227 def heads(self, start=None):
3228 if start is None:
3228 if start is None:
3229 cl = self.changelog
3229 cl = self.changelog
3230 headrevs = reversed(cl.headrevs())
3230 headrevs = reversed(cl.headrevs())
3231 return [cl.node(rev) for rev in headrevs]
3231 return [cl.node(rev) for rev in headrevs]
3232
3232
3233 heads = self.changelog.heads(start)
3233 heads = self.changelog.heads(start)
3234 # sort the output in rev descending order
3234 # sort the output in rev descending order
3235 return sorted(heads, key=self.changelog.rev, reverse=True)
3235 return sorted(heads, key=self.changelog.rev, reverse=True)
3236
3236
3237 def branchheads(self, branch=None, start=None, closed=False):
3237 def branchheads(self, branch=None, start=None, closed=False):
3238 """return a (possibly filtered) list of heads for the given branch
3238 """return a (possibly filtered) list of heads for the given branch
3239
3239
3240 Heads are returned in topological order, from newest to oldest.
3240 Heads are returned in topological order, from newest to oldest.
3241 If branch is None, use the dirstate branch.
3241 If branch is None, use the dirstate branch.
3242 If start is not None, return only heads reachable from start.
3242 If start is not None, return only heads reachable from start.
3243 If closed is True, return heads that are marked as closed as well.
3243 If closed is True, return heads that are marked as closed as well.
3244 """
3244 """
3245 if branch is None:
3245 if branch is None:
3246 branch = self[None].branch()
3246 branch = self[None].branch()
3247 branches = self.branchmap()
3247 branches = self.branchmap()
3248 if not branches.hasbranch(branch):
3248 if not branches.hasbranch(branch):
3249 return []
3249 return []
3250 # the cache returns heads ordered lowest to highest
3250 # the cache returns heads ordered lowest to highest
3251 bheads = list(reversed(branches.branchheads(branch, closed=closed)))
3251 bheads = list(reversed(branches.branchheads(branch, closed=closed)))
3252 if start is not None:
3252 if start is not None:
3253 # filter out the heads that cannot be reached from startrev
3253 # filter out the heads that cannot be reached from startrev
3254 fbheads = set(self.changelog.nodesbetween([start], bheads)[2])
3254 fbheads = set(self.changelog.nodesbetween([start], bheads)[2])
3255 bheads = [h for h in bheads if h in fbheads]
3255 bheads = [h for h in bheads if h in fbheads]
3256 return bheads
3256 return bheads
3257
3257
3258 def branches(self, nodes):
3258 def branches(self, nodes):
3259 if not nodes:
3259 if not nodes:
3260 nodes = [self.changelog.tip()]
3260 nodes = [self.changelog.tip()]
3261 b = []
3261 b = []
3262 for n in nodes:
3262 for n in nodes:
3263 t = n
3263 t = n
3264 while True:
3264 while True:
3265 p = self.changelog.parents(n)
3265 p = self.changelog.parents(n)
3266 if p[1] != nullid or p[0] == nullid:
3266 if p[1] != nullid or p[0] == nullid:
3267 b.append((t, n, p[0], p[1]))
3267 b.append((t, n, p[0], p[1]))
3268 break
3268 break
3269 n = p[0]
3269 n = p[0]
3270 return b
3270 return b
3271
3271
3272 def between(self, pairs):
3272 def between(self, pairs):
3273 r = []
3273 r = []
3274
3274
3275 for top, bottom in pairs:
3275 for top, bottom in pairs:
3276 n, l, i = top, [], 0
3276 n, l, i = top, [], 0
3277 f = 1
3277 f = 1
3278
3278
3279 while n != bottom and n != nullid:
3279 while n != bottom and n != nullid:
3280 p = self.changelog.parents(n)[0]
3280 p = self.changelog.parents(n)[0]
3281 if i == f:
3281 if i == f:
3282 l.append(n)
3282 l.append(n)
3283 f = f * 2
3283 f = f * 2
3284 n = p
3284 n = p
3285 i += 1
3285 i += 1
3286
3286
3287 r.append(l)
3287 r.append(l)
3288
3288
3289 return r
3289 return r
3290
3290
3291 def checkpush(self, pushop):
3291 def checkpush(self, pushop):
3292 """Extensions can override this function if additional checks have
3292 """Extensions can override this function if additional checks have
3293 to be performed before pushing, or call it if they override push
3293 to be performed before pushing, or call it if they override push
3294 command.
3294 command.
3295 """
3295 """
3296
3296
3297 @unfilteredpropertycache
3297 @unfilteredpropertycache
3298 def prepushoutgoinghooks(self):
3298 def prepushoutgoinghooks(self):
3299 """Return util.hooks consists of a pushop with repo, remote, outgoing
3299 """Return util.hooks consists of a pushop with repo, remote, outgoing
3300 methods, which are called before pushing changesets.
3300 methods, which are called before pushing changesets.
3301 """
3301 """
3302 return util.hooks()
3302 return util.hooks()
3303
3303
3304 def pushkey(self, namespace, key, old, new):
3304 def pushkey(self, namespace, key, old, new):
3305 try:
3305 try:
3306 tr = self.currenttransaction()
3306 tr = self.currenttransaction()
3307 hookargs = {}
3307 hookargs = {}
3308 if tr is not None:
3308 if tr is not None:
3309 hookargs.update(tr.hookargs)
3309 hookargs.update(tr.hookargs)
3310 hookargs = pycompat.strkwargs(hookargs)
3310 hookargs = pycompat.strkwargs(hookargs)
3311 hookargs['namespace'] = namespace
3311 hookargs['namespace'] = namespace
3312 hookargs['key'] = key
3312 hookargs['key'] = key
3313 hookargs['old'] = old
3313 hookargs['old'] = old
3314 hookargs['new'] = new
3314 hookargs['new'] = new
3315 self.hook(b'prepushkey', throw=True, **hookargs)
3315 self.hook(b'prepushkey', throw=True, **hookargs)
3316 except error.HookAbort as exc:
3316 except error.HookAbort as exc:
3317 self.ui.write_err(_(b"pushkey-abort: %s\n") % exc)
3317 self.ui.write_err(_(b"pushkey-abort: %s\n") % exc)
3318 if exc.hint:
3318 if exc.hint:
3319 self.ui.write_err(_(b"(%s)\n") % exc.hint)
3319 self.ui.write_err(_(b"(%s)\n") % exc.hint)
3320 return False
3320 return False
3321 self.ui.debug(b'pushing key for "%s:%s"\n' % (namespace, key))
3321 self.ui.debug(b'pushing key for "%s:%s"\n' % (namespace, key))
3322 ret = pushkey.push(self, namespace, key, old, new)
3322 ret = pushkey.push(self, namespace, key, old, new)
3323
3323
3324 def runhook(unused_success):
3324 def runhook(unused_success):
3325 self.hook(
3325 self.hook(
3326 b'pushkey',
3326 b'pushkey',
3327 namespace=namespace,
3327 namespace=namespace,
3328 key=key,
3328 key=key,
3329 old=old,
3329 old=old,
3330 new=new,
3330 new=new,
3331 ret=ret,
3331 ret=ret,
3332 )
3332 )
3333
3333
3334 self._afterlock(runhook)
3334 self._afterlock(runhook)
3335 return ret
3335 return ret
3336
3336
3337 def listkeys(self, namespace):
3337 def listkeys(self, namespace):
3338 self.hook(b'prelistkeys', throw=True, namespace=namespace)
3338 self.hook(b'prelistkeys', throw=True, namespace=namespace)
3339 self.ui.debug(b'listing keys for "%s"\n' % namespace)
3339 self.ui.debug(b'listing keys for "%s"\n' % namespace)
3340 values = pushkey.list(self, namespace)
3340 values = pushkey.list(self, namespace)
3341 self.hook(b'listkeys', namespace=namespace, values=values)
3341 self.hook(b'listkeys', namespace=namespace, values=values)
3342 return values
3342 return values
3343
3343
3344 def debugwireargs(self, one, two, three=None, four=None, five=None):
3344 def debugwireargs(self, one, two, three=None, four=None, five=None):
3345 '''used to test argument passing over the wire'''
3345 '''used to test argument passing over the wire'''
3346 return b"%s %s %s %s %s" % (
3346 return b"%s %s %s %s %s" % (
3347 one,
3347 one,
3348 two,
3348 two,
3349 pycompat.bytestr(three),
3349 pycompat.bytestr(three),
3350 pycompat.bytestr(four),
3350 pycompat.bytestr(four),
3351 pycompat.bytestr(five),
3351 pycompat.bytestr(five),
3352 )
3352 )
3353
3353
3354 def savecommitmessage(self, text):
3354 def savecommitmessage(self, text):
3355 fp = self.vfs(b'last-message.txt', b'wb')
3355 fp = self.vfs(b'last-message.txt', b'wb')
3356 try:
3356 try:
3357 fp.write(text)
3357 fp.write(text)
3358 finally:
3358 finally:
3359 fp.close()
3359 fp.close()
3360 return self.pathto(fp.name[len(self.root) + 1 :])
3360 return self.pathto(fp.name[len(self.root) + 1 :])
3361
3361
3362 def register_wanted_sidedata(self, category):
3362 def register_wanted_sidedata(self, category):
3363 self._wanted_sidedata.add(pycompat.bytestr(category))
3363 self._wanted_sidedata.add(pycompat.bytestr(category))
3364
3364
3365 def register_sidedata_computer(self, kind, category, keys, computer):
3365 def register_sidedata_computer(self, kind, category, keys, computer):
3366 if kind not in (b"changelog", b"manifest", b"filelog"):
3366 if kind not in (b"changelog", b"manifest", b"filelog"):
3367 msg = _(b"unexpected revlog kind '%s'.")
3367 msg = _(b"unexpected revlog kind '%s'.")
3368 raise error.ProgrammingError(msg % kind)
3368 raise error.ProgrammingError(msg % kind)
3369 category = pycompat.bytestr(category)
3369 category = pycompat.bytestr(category)
3370 if category in self._sidedata_computers.get(kind, []):
3370 if category in self._sidedata_computers.get(kind, []):
3371 msg = _(
3371 msg = _(
3372 b"cannot register a sidedata computer twice for category '%s'."
3372 b"cannot register a sidedata computer twice for category '%s'."
3373 )
3373 )
3374 raise error.ProgrammingError(msg % category)
3374 raise error.ProgrammingError(msg % category)
3375 self._sidedata_computers.setdefault(kind, {})
3375 self._sidedata_computers.setdefault(kind, {})
3376 self._sidedata_computers[kind][category] = (keys, computer)
3376 self._sidedata_computers[kind][category] = (keys, computer)
3377
3377
3378
3378
3379 # used to avoid circular references so destructors work
3379 # used to avoid circular references so destructors work
3380 def aftertrans(files):
3380 def aftertrans(files):
3381 renamefiles = [tuple(t) for t in files]
3381 renamefiles = [tuple(t) for t in files]
3382
3382
3383 def a():
3383 def a():
3384 for vfs, src, dest in renamefiles:
3384 for vfs, src, dest in renamefiles:
3385 # if src and dest refer to a same file, vfs.rename is a no-op,
3385 # if src and dest refer to a same file, vfs.rename is a no-op,
3386 # leaving both src and dest on disk. delete dest to make sure
3386 # leaving both src and dest on disk. delete dest to make sure
3387 # the rename couldn't be such a no-op.
3387 # the rename couldn't be such a no-op.
3388 vfs.tryunlink(dest)
3388 vfs.tryunlink(dest)
3389 try:
3389 try:
3390 vfs.rename(src, dest)
3390 vfs.rename(src, dest)
3391 except OSError: # journal file does not yet exist
3391 except OSError: # journal file does not yet exist
3392 pass
3392 pass
3393
3393
3394 return a
3394 return a
3395
3395
3396
3396
3397 def undoname(fn):
3397 def undoname(fn):
3398 base, name = os.path.split(fn)
3398 base, name = os.path.split(fn)
3399 assert name.startswith(b'journal')
3399 assert name.startswith(b'journal')
3400 return os.path.join(base, name.replace(b'journal', b'undo', 1))
3400 return os.path.join(base, name.replace(b'journal', b'undo', 1))
3401
3401
3402
3402
3403 def instance(ui, path, create, intents=None, createopts=None):
3403 def instance(ui, path, create, intents=None, createopts=None):
3404 localpath = util.urllocalpath(path)
3404 localpath = util.urllocalpath(path)
3405 if create:
3405 if create:
3406 createrepository(ui, localpath, createopts=createopts)
3406 createrepository(ui, localpath, createopts=createopts)
3407
3407
3408 return makelocalrepository(ui, localpath, intents=intents)
3408 return makelocalrepository(ui, localpath, intents=intents)
3409
3409
3410
3410
3411 def islocal(path):
3411 def islocal(path):
3412 return True
3412 return True
3413
3413
3414
3414
3415 def defaultcreateopts(ui, createopts=None):
3415 def defaultcreateopts(ui, createopts=None):
3416 """Populate the default creation options for a repository.
3416 """Populate the default creation options for a repository.
3417
3417
3418 A dictionary of explicitly requested creation options can be passed
3418 A dictionary of explicitly requested creation options can be passed
3419 in. Missing keys will be populated.
3419 in. Missing keys will be populated.
3420 """
3420 """
3421 createopts = dict(createopts or {})
3421 createopts = dict(createopts or {})
3422
3422
3423 if b'backend' not in createopts:
3423 if b'backend' not in createopts:
3424 # experimental config: storage.new-repo-backend
3424 # experimental config: storage.new-repo-backend
3425 createopts[b'backend'] = ui.config(b'storage', b'new-repo-backend')
3425 createopts[b'backend'] = ui.config(b'storage', b'new-repo-backend')
3426
3426
3427 return createopts
3427 return createopts
3428
3428
3429
3429
3430 def newreporequirements(ui, createopts):
3430 def newreporequirements(ui, createopts):
3431 """Determine the set of requirements for a new local repository.
3431 """Determine the set of requirements for a new local repository.
3432
3432
3433 Extensions can wrap this function to specify custom requirements for
3433 Extensions can wrap this function to specify custom requirements for
3434 new repositories.
3434 new repositories.
3435 """
3435 """
3436 # If the repo is being created from a shared repository, we copy
3436 # If the repo is being created from a shared repository, we copy
3437 # its requirements.
3437 # its requirements.
3438 if b'sharedrepo' in createopts:
3438 if b'sharedrepo' in createopts:
3439 requirements = set(createopts[b'sharedrepo'].requirements)
3439 requirements = set(createopts[b'sharedrepo'].requirements)
3440 if createopts.get(b'sharedrelative'):
3440 if createopts.get(b'sharedrelative'):
3441 requirements.add(requirementsmod.RELATIVE_SHARED_REQUIREMENT)
3441 requirements.add(requirementsmod.RELATIVE_SHARED_REQUIREMENT)
3442 else:
3442 else:
3443 requirements.add(requirementsmod.SHARED_REQUIREMENT)
3443 requirements.add(requirementsmod.SHARED_REQUIREMENT)
3444
3444
3445 return requirements
3445 return requirements
3446
3446
3447 if b'backend' not in createopts:
3447 if b'backend' not in createopts:
3448 raise error.ProgrammingError(
3448 raise error.ProgrammingError(
3449 b'backend key not present in createopts; '
3449 b'backend key not present in createopts; '
3450 b'was defaultcreateopts() called?'
3450 b'was defaultcreateopts() called?'
3451 )
3451 )
3452
3452
3453 if createopts[b'backend'] != b'revlogv1':
3453 if createopts[b'backend'] != b'revlogv1':
3454 raise error.Abort(
3454 raise error.Abort(
3455 _(
3455 _(
3456 b'unable to determine repository requirements for '
3456 b'unable to determine repository requirements for '
3457 b'storage backend: %s'
3457 b'storage backend: %s'
3458 )
3458 )
3459 % createopts[b'backend']
3459 % createopts[b'backend']
3460 )
3460 )
3461
3461
3462 requirements = {requirementsmod.REVLOGV1_REQUIREMENT}
3462 requirements = {requirementsmod.REVLOGV1_REQUIREMENT}
3463 if ui.configbool(b'format', b'usestore'):
3463 if ui.configbool(b'format', b'usestore'):
3464 requirements.add(requirementsmod.STORE_REQUIREMENT)
3464 requirements.add(requirementsmod.STORE_REQUIREMENT)
3465 if ui.configbool(b'format', b'usefncache'):
3465 if ui.configbool(b'format', b'usefncache'):
3466 requirements.add(requirementsmod.FNCACHE_REQUIREMENT)
3466 requirements.add(requirementsmod.FNCACHE_REQUIREMENT)
3467 if ui.configbool(b'format', b'dotencode'):
3467 if ui.configbool(b'format', b'dotencode'):
3468 requirements.add(requirementsmod.DOTENCODE_REQUIREMENT)
3468 requirements.add(requirementsmod.DOTENCODE_REQUIREMENT)
3469
3469
3470 compengines = ui.configlist(b'format', b'revlog-compression')
3470 compengines = ui.configlist(b'format', b'revlog-compression')
3471 for compengine in compengines:
3471 for compengine in compengines:
3472 if compengine in util.compengines:
3472 if compengine in util.compengines:
3473 break
3473 engine = util.compengines[compengine]
3474 if engine.available() and engine.revlogheader():
3475 break
3474 else:
3476 else:
3475 raise error.Abort(
3477 raise error.Abort(
3476 _(
3478 _(
3477 b'compression engines %s defined by '
3479 b'compression engines %s defined by '
3478 b'format.revlog-compression not available'
3480 b'format.revlog-compression not available'
3479 )
3481 )
3480 % b', '.join(b'"%s"' % e for e in compengines),
3482 % b', '.join(b'"%s"' % e for e in compengines),
3481 hint=_(
3483 hint=_(
3482 b'run "hg debuginstall" to list available '
3484 b'run "hg debuginstall" to list available '
3483 b'compression engines'
3485 b'compression engines'
3484 ),
3486 ),
3485 )
3487 )
3486
3488
3487 # zlib is the historical default and doesn't need an explicit requirement.
3489 # zlib is the historical default and doesn't need an explicit requirement.
3488 if compengine == b'zstd':
3490 if compengine == b'zstd':
3489 requirements.add(b'revlog-compression-zstd')
3491 requirements.add(b'revlog-compression-zstd')
3490 elif compengine != b'zlib':
3492 elif compengine != b'zlib':
3491 requirements.add(b'exp-compression-%s' % compengine)
3493 requirements.add(b'exp-compression-%s' % compengine)
3492
3494
3493 if scmutil.gdinitconfig(ui):
3495 if scmutil.gdinitconfig(ui):
3494 requirements.add(requirementsmod.GENERALDELTA_REQUIREMENT)
3496 requirements.add(requirementsmod.GENERALDELTA_REQUIREMENT)
3495 if ui.configbool(b'format', b'sparse-revlog'):
3497 if ui.configbool(b'format', b'sparse-revlog'):
3496 requirements.add(requirementsmod.SPARSEREVLOG_REQUIREMENT)
3498 requirements.add(requirementsmod.SPARSEREVLOG_REQUIREMENT)
3497
3499
3498 # experimental config: format.exp-use-side-data
3500 # experimental config: format.exp-use-side-data
3499 if ui.configbool(b'format', b'exp-use-side-data'):
3501 if ui.configbool(b'format', b'exp-use-side-data'):
3500 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3502 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3501 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3503 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3502 requirements.add(requirementsmod.SIDEDATA_REQUIREMENT)
3504 requirements.add(requirementsmod.SIDEDATA_REQUIREMENT)
3503 # experimental config: format.exp-use-copies-side-data-changeset
3505 # experimental config: format.exp-use-copies-side-data-changeset
3504 if ui.configbool(b'format', b'exp-use-copies-side-data-changeset'):
3506 if ui.configbool(b'format', b'exp-use-copies-side-data-changeset'):
3505 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3507 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3506 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3508 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3507 requirements.add(requirementsmod.SIDEDATA_REQUIREMENT)
3509 requirements.add(requirementsmod.SIDEDATA_REQUIREMENT)
3508 requirements.add(requirementsmod.COPIESSDC_REQUIREMENT)
3510 requirements.add(requirementsmod.COPIESSDC_REQUIREMENT)
3509 if ui.configbool(b'experimental', b'treemanifest'):
3511 if ui.configbool(b'experimental', b'treemanifest'):
3510 requirements.add(requirementsmod.TREEMANIFEST_REQUIREMENT)
3512 requirements.add(requirementsmod.TREEMANIFEST_REQUIREMENT)
3511
3513
3512 revlogv2 = ui.config(b'experimental', b'revlogv2')
3514 revlogv2 = ui.config(b'experimental', b'revlogv2')
3513 if revlogv2 == b'enable-unstable-format-and-corrupt-my-data':
3515 if revlogv2 == b'enable-unstable-format-and-corrupt-my-data':
3514 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3516 requirements.discard(requirementsmod.REVLOGV1_REQUIREMENT)
3515 # generaldelta is implied by revlogv2.
3517 # generaldelta is implied by revlogv2.
3516 requirements.discard(requirementsmod.GENERALDELTA_REQUIREMENT)
3518 requirements.discard(requirementsmod.GENERALDELTA_REQUIREMENT)
3517 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3519 requirements.add(requirementsmod.REVLOGV2_REQUIREMENT)
3518 # experimental config: format.internal-phase
3520 # experimental config: format.internal-phase
3519 if ui.configbool(b'format', b'internal-phase'):
3521 if ui.configbool(b'format', b'internal-phase'):
3520 requirements.add(requirementsmod.INTERNAL_PHASE_REQUIREMENT)
3522 requirements.add(requirementsmod.INTERNAL_PHASE_REQUIREMENT)
3521
3523
3522 if createopts.get(b'narrowfiles'):
3524 if createopts.get(b'narrowfiles'):
3523 requirements.add(requirementsmod.NARROW_REQUIREMENT)
3525 requirements.add(requirementsmod.NARROW_REQUIREMENT)
3524
3526
3525 if createopts.get(b'lfs'):
3527 if createopts.get(b'lfs'):
3526 requirements.add(b'lfs')
3528 requirements.add(b'lfs')
3527
3529
3528 if ui.configbool(b'format', b'bookmarks-in-store'):
3530 if ui.configbool(b'format', b'bookmarks-in-store'):
3529 requirements.add(bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT)
3531 requirements.add(bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT)
3530
3532
3531 if ui.configbool(b'format', b'use-persistent-nodemap'):
3533 if ui.configbool(b'format', b'use-persistent-nodemap'):
3532 requirements.add(requirementsmod.NODEMAP_REQUIREMENT)
3534 requirements.add(requirementsmod.NODEMAP_REQUIREMENT)
3533
3535
3534 # if share-safe is enabled, let's create the new repository with the new
3536 # if share-safe is enabled, let's create the new repository with the new
3535 # requirement
3537 # requirement
3536 if ui.configbool(b'format', b'use-share-safe'):
3538 if ui.configbool(b'format', b'use-share-safe'):
3537 requirements.add(requirementsmod.SHARESAFE_REQUIREMENT)
3539 requirements.add(requirementsmod.SHARESAFE_REQUIREMENT)
3538
3540
3539 return requirements
3541 return requirements
3540
3542
3541
3543
3542 def checkrequirementscompat(ui, requirements):
3544 def checkrequirementscompat(ui, requirements):
3543 """Checks compatibility of repository requirements enabled and disabled.
3545 """Checks compatibility of repository requirements enabled and disabled.
3544
3546
3545 Returns a set of requirements which needs to be dropped because dependend
3547 Returns a set of requirements which needs to be dropped because dependend
3546 requirements are not enabled. Also warns users about it"""
3548 requirements are not enabled. Also warns users about it"""
3547
3549
3548 dropped = set()
3550 dropped = set()
3549
3551
3550 if requirementsmod.STORE_REQUIREMENT not in requirements:
3552 if requirementsmod.STORE_REQUIREMENT not in requirements:
3551 if bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT in requirements:
3553 if bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT in requirements:
3552 ui.warn(
3554 ui.warn(
3553 _(
3555 _(
3554 b'ignoring enabled \'format.bookmarks-in-store\' config '
3556 b'ignoring enabled \'format.bookmarks-in-store\' config '
3555 b'beacuse it is incompatible with disabled '
3557 b'beacuse it is incompatible with disabled '
3556 b'\'format.usestore\' config\n'
3558 b'\'format.usestore\' config\n'
3557 )
3559 )
3558 )
3560 )
3559 dropped.add(bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT)
3561 dropped.add(bookmarks.BOOKMARKS_IN_STORE_REQUIREMENT)
3560
3562
3561 if (
3563 if (
3562 requirementsmod.SHARED_REQUIREMENT in requirements
3564 requirementsmod.SHARED_REQUIREMENT in requirements
3563 or requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements
3565 or requirementsmod.RELATIVE_SHARED_REQUIREMENT in requirements
3564 ):
3566 ):
3565 raise error.Abort(
3567 raise error.Abort(
3566 _(
3568 _(
3567 b"cannot create shared repository as source was created"
3569 b"cannot create shared repository as source was created"
3568 b" with 'format.usestore' config disabled"
3570 b" with 'format.usestore' config disabled"
3569 )
3571 )
3570 )
3572 )
3571
3573
3572 if requirementsmod.SHARESAFE_REQUIREMENT in requirements:
3574 if requirementsmod.SHARESAFE_REQUIREMENT in requirements:
3573 ui.warn(
3575 ui.warn(
3574 _(
3576 _(
3575 b"ignoring enabled 'format.use-share-safe' config because "
3577 b"ignoring enabled 'format.use-share-safe' config because "
3576 b"it is incompatible with disabled 'format.usestore'"
3578 b"it is incompatible with disabled 'format.usestore'"
3577 b" config\n"
3579 b" config\n"
3578 )
3580 )
3579 )
3581 )
3580 dropped.add(requirementsmod.SHARESAFE_REQUIREMENT)
3582 dropped.add(requirementsmod.SHARESAFE_REQUIREMENT)
3581
3583
3582 return dropped
3584 return dropped
3583
3585
3584
3586
3585 def filterknowncreateopts(ui, createopts):
3587 def filterknowncreateopts(ui, createopts):
3586 """Filters a dict of repo creation options against options that are known.
3588 """Filters a dict of repo creation options against options that are known.
3587
3589
3588 Receives a dict of repo creation options and returns a dict of those
3590 Receives a dict of repo creation options and returns a dict of those
3589 options that we don't know how to handle.
3591 options that we don't know how to handle.
3590
3592
3591 This function is called as part of repository creation. If the
3593 This function is called as part of repository creation. If the
3592 returned dict contains any items, repository creation will not
3594 returned dict contains any items, repository creation will not
3593 be allowed, as it means there was a request to create a repository
3595 be allowed, as it means there was a request to create a repository
3594 with options not recognized by loaded code.
3596 with options not recognized by loaded code.
3595
3597
3596 Extensions can wrap this function to filter out creation options
3598 Extensions can wrap this function to filter out creation options
3597 they know how to handle.
3599 they know how to handle.
3598 """
3600 """
3599 known = {
3601 known = {
3600 b'backend',
3602 b'backend',
3601 b'lfs',
3603 b'lfs',
3602 b'narrowfiles',
3604 b'narrowfiles',
3603 b'sharedrepo',
3605 b'sharedrepo',
3604 b'sharedrelative',
3606 b'sharedrelative',
3605 b'shareditems',
3607 b'shareditems',
3606 b'shallowfilestore',
3608 b'shallowfilestore',
3607 }
3609 }
3608
3610
3609 return {k: v for k, v in createopts.items() if k not in known}
3611 return {k: v for k, v in createopts.items() if k not in known}
3610
3612
3611
3613
3612 def createrepository(ui, path, createopts=None):
3614 def createrepository(ui, path, createopts=None):
3613 """Create a new repository in a vfs.
3615 """Create a new repository in a vfs.
3614
3616
3615 ``path`` path to the new repo's working directory.
3617 ``path`` path to the new repo's working directory.
3616 ``createopts`` options for the new repository.
3618 ``createopts`` options for the new repository.
3617
3619
3618 The following keys for ``createopts`` are recognized:
3620 The following keys for ``createopts`` are recognized:
3619
3621
3620 backend
3622 backend
3621 The storage backend to use.
3623 The storage backend to use.
3622 lfs
3624 lfs
3623 Repository will be created with ``lfs`` requirement. The lfs extension
3625 Repository will be created with ``lfs`` requirement. The lfs extension
3624 will automatically be loaded when the repository is accessed.
3626 will automatically be loaded when the repository is accessed.
3625 narrowfiles
3627 narrowfiles
3626 Set up repository to support narrow file storage.
3628 Set up repository to support narrow file storage.
3627 sharedrepo
3629 sharedrepo
3628 Repository object from which storage should be shared.
3630 Repository object from which storage should be shared.
3629 sharedrelative
3631 sharedrelative
3630 Boolean indicating if the path to the shared repo should be
3632 Boolean indicating if the path to the shared repo should be
3631 stored as relative. By default, the pointer to the "parent" repo
3633 stored as relative. By default, the pointer to the "parent" repo
3632 is stored as an absolute path.
3634 is stored as an absolute path.
3633 shareditems
3635 shareditems
3634 Set of items to share to the new repository (in addition to storage).
3636 Set of items to share to the new repository (in addition to storage).
3635 shallowfilestore
3637 shallowfilestore
3636 Indicates that storage for files should be shallow (not all ancestor
3638 Indicates that storage for files should be shallow (not all ancestor
3637 revisions are known).
3639 revisions are known).
3638 """
3640 """
3639 createopts = defaultcreateopts(ui, createopts=createopts)
3641 createopts = defaultcreateopts(ui, createopts=createopts)
3640
3642
3641 unknownopts = filterknowncreateopts(ui, createopts)
3643 unknownopts = filterknowncreateopts(ui, createopts)
3642
3644
3643 if not isinstance(unknownopts, dict):
3645 if not isinstance(unknownopts, dict):
3644 raise error.ProgrammingError(
3646 raise error.ProgrammingError(
3645 b'filterknowncreateopts() did not return a dict'
3647 b'filterknowncreateopts() did not return a dict'
3646 )
3648 )
3647
3649
3648 if unknownopts:
3650 if unknownopts:
3649 raise error.Abort(
3651 raise error.Abort(
3650 _(
3652 _(
3651 b'unable to create repository because of unknown '
3653 b'unable to create repository because of unknown '
3652 b'creation option: %s'
3654 b'creation option: %s'
3653 )
3655 )
3654 % b', '.join(sorted(unknownopts)),
3656 % b', '.join(sorted(unknownopts)),
3655 hint=_(b'is a required extension not loaded?'),
3657 hint=_(b'is a required extension not loaded?'),
3656 )
3658 )
3657
3659
3658 requirements = newreporequirements(ui, createopts=createopts)
3660 requirements = newreporequirements(ui, createopts=createopts)
3659 requirements -= checkrequirementscompat(ui, requirements)
3661 requirements -= checkrequirementscompat(ui, requirements)
3660
3662
3661 wdirvfs = vfsmod.vfs(path, expandpath=True, realpath=True)
3663 wdirvfs = vfsmod.vfs(path, expandpath=True, realpath=True)
3662
3664
3663 hgvfs = vfsmod.vfs(wdirvfs.join(b'.hg'))
3665 hgvfs = vfsmod.vfs(wdirvfs.join(b'.hg'))
3664 if hgvfs.exists():
3666 if hgvfs.exists():
3665 raise error.RepoError(_(b'repository %s already exists') % path)
3667 raise error.RepoError(_(b'repository %s already exists') % path)
3666
3668
3667 if b'sharedrepo' in createopts:
3669 if b'sharedrepo' in createopts:
3668 sharedpath = createopts[b'sharedrepo'].sharedpath
3670 sharedpath = createopts[b'sharedrepo'].sharedpath
3669
3671
3670 if createopts.get(b'sharedrelative'):
3672 if createopts.get(b'sharedrelative'):
3671 try:
3673 try:
3672 sharedpath = os.path.relpath(sharedpath, hgvfs.base)
3674 sharedpath = os.path.relpath(sharedpath, hgvfs.base)
3673 except (IOError, ValueError) as e:
3675 except (IOError, ValueError) as e:
3674 # ValueError is raised on Windows if the drive letters differ
3676 # ValueError is raised on Windows if the drive letters differ
3675 # on each path.
3677 # on each path.
3676 raise error.Abort(
3678 raise error.Abort(
3677 _(b'cannot calculate relative path'),
3679 _(b'cannot calculate relative path'),
3678 hint=stringutil.forcebytestr(e),
3680 hint=stringutil.forcebytestr(e),
3679 )
3681 )
3680
3682
3681 if not wdirvfs.exists():
3683 if not wdirvfs.exists():
3682 wdirvfs.makedirs()
3684 wdirvfs.makedirs()
3683
3685
3684 hgvfs.makedir(notindexed=True)
3686 hgvfs.makedir(notindexed=True)
3685 if b'sharedrepo' not in createopts:
3687 if b'sharedrepo' not in createopts:
3686 hgvfs.mkdir(b'cache')
3688 hgvfs.mkdir(b'cache')
3687 hgvfs.mkdir(b'wcache')
3689 hgvfs.mkdir(b'wcache')
3688
3690
3689 has_store = requirementsmod.STORE_REQUIREMENT in requirements
3691 has_store = requirementsmod.STORE_REQUIREMENT in requirements
3690 if has_store and b'sharedrepo' not in createopts:
3692 if has_store and b'sharedrepo' not in createopts:
3691 hgvfs.mkdir(b'store')
3693 hgvfs.mkdir(b'store')
3692
3694
3693 # We create an invalid changelog outside the store so very old
3695 # We create an invalid changelog outside the store so very old
3694 # Mercurial versions (which didn't know about the requirements
3696 # Mercurial versions (which didn't know about the requirements
3695 # file) encounter an error on reading the changelog. This
3697 # file) encounter an error on reading the changelog. This
3696 # effectively locks out old clients and prevents them from
3698 # effectively locks out old clients and prevents them from
3697 # mucking with a repo in an unknown format.
3699 # mucking with a repo in an unknown format.
3698 #
3700 #
3699 # The revlog header has version 65535, which won't be recognized by
3701 # The revlog header has version 65535, which won't be recognized by
3700 # such old clients.
3702 # such old clients.
3701 hgvfs.append(
3703 hgvfs.append(
3702 b'00changelog.i',
3704 b'00changelog.i',
3703 b'\0\0\xFF\xFF dummy changelog to prevent using the old repo '
3705 b'\0\0\xFF\xFF dummy changelog to prevent using the old repo '
3704 b'layout',
3706 b'layout',
3705 )
3707 )
3706
3708
3707 # Filter the requirements into working copy and store ones
3709 # Filter the requirements into working copy and store ones
3708 wcreq, storereq = scmutil.filterrequirements(requirements)
3710 wcreq, storereq = scmutil.filterrequirements(requirements)
3709 # write working copy ones
3711 # write working copy ones
3710 scmutil.writerequires(hgvfs, wcreq)
3712 scmutil.writerequires(hgvfs, wcreq)
3711 # If there are store requirements and the current repository
3713 # If there are store requirements and the current repository
3712 # is not a shared one, write stored requirements
3714 # is not a shared one, write stored requirements
3713 # For new shared repository, we don't need to write the store
3715 # For new shared repository, we don't need to write the store
3714 # requirements as they are already present in store requires
3716 # requirements as they are already present in store requires
3715 if storereq and b'sharedrepo' not in createopts:
3717 if storereq and b'sharedrepo' not in createopts:
3716 storevfs = vfsmod.vfs(hgvfs.join(b'store'), cacheaudited=True)
3718 storevfs = vfsmod.vfs(hgvfs.join(b'store'), cacheaudited=True)
3717 scmutil.writerequires(storevfs, storereq)
3719 scmutil.writerequires(storevfs, storereq)
3718
3720
3719 # Write out file telling readers where to find the shared store.
3721 # Write out file telling readers where to find the shared store.
3720 if b'sharedrepo' in createopts:
3722 if b'sharedrepo' in createopts:
3721 hgvfs.write(b'sharedpath', sharedpath)
3723 hgvfs.write(b'sharedpath', sharedpath)
3722
3724
3723 if createopts.get(b'shareditems'):
3725 if createopts.get(b'shareditems'):
3724 shared = b'\n'.join(sorted(createopts[b'shareditems'])) + b'\n'
3726 shared = b'\n'.join(sorted(createopts[b'shareditems'])) + b'\n'
3725 hgvfs.write(b'shared', shared)
3727 hgvfs.write(b'shared', shared)
3726
3728
3727
3729
3728 def poisonrepository(repo):
3730 def poisonrepository(repo):
3729 """Poison a repository instance so it can no longer be used."""
3731 """Poison a repository instance so it can no longer be used."""
3730 # Perform any cleanup on the instance.
3732 # Perform any cleanup on the instance.
3731 repo.close()
3733 repo.close()
3732
3734
3733 # Our strategy is to replace the type of the object with one that
3735 # Our strategy is to replace the type of the object with one that
3734 # has all attribute lookups result in error.
3736 # has all attribute lookups result in error.
3735 #
3737 #
3736 # But we have to allow the close() method because some constructors
3738 # But we have to allow the close() method because some constructors
3737 # of repos call close() on repo references.
3739 # of repos call close() on repo references.
3738 class poisonedrepository(object):
3740 class poisonedrepository(object):
3739 def __getattribute__(self, item):
3741 def __getattribute__(self, item):
3740 if item == 'close':
3742 if item == 'close':
3741 return object.__getattribute__(self, item)
3743 return object.__getattribute__(self, item)
3742
3744
3743 raise error.ProgrammingError(
3745 raise error.ProgrammingError(
3744 b'repo instances should not be used after unshare'
3746 b'repo instances should not be used after unshare'
3745 )
3747 )
3746
3748
3747 def close(self):
3749 def close(self):
3748 pass
3750 pass
3749
3751
3750 # We may have a repoview, which intercepts __setattr__. So be sure
3752 # We may have a repoview, which intercepts __setattr__. So be sure
3751 # we operate at the lowest level possible.
3753 # we operate at the lowest level possible.
3752 object.__setattr__(repo, '__class__', poisonedrepository)
3754 object.__setattr__(repo, '__class__', poisonedrepository)
@@ -1,1019 +1,1021 b''
1 # upgrade.py - functions for in place upgrade of Mercurial repository
1 # upgrade.py - functions for in place upgrade of Mercurial repository
2 #
2 #
3 # Copyright (c) 2016-present, Gregory Szorc
3 # Copyright (c) 2016-present, Gregory Szorc
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 from __future__ import absolute_import
8 from __future__ import absolute_import
9
9
10 from ..i18n import _
10 from ..i18n import _
11 from .. import (
11 from .. import (
12 error,
12 error,
13 localrepo,
13 localrepo,
14 requirements,
14 requirements,
15 revlog,
15 revlog,
16 util,
16 util,
17 )
17 )
18
18
19 from ..utils import compression
19 from ..utils import compression
20
20
21 # list of requirements that request a clone of all revlog if added/removed
21 # list of requirements that request a clone of all revlog if added/removed
22 RECLONES_REQUIREMENTS = {
22 RECLONES_REQUIREMENTS = {
23 requirements.GENERALDELTA_REQUIREMENT,
23 requirements.GENERALDELTA_REQUIREMENT,
24 requirements.SPARSEREVLOG_REQUIREMENT,
24 requirements.SPARSEREVLOG_REQUIREMENT,
25 }
25 }
26
26
27
27
28 def preservedrequirements(repo):
28 def preservedrequirements(repo):
29 return set()
29 return set()
30
30
31
31
32 FORMAT_VARIANT = b'deficiency'
32 FORMAT_VARIANT = b'deficiency'
33 OPTIMISATION = b'optimization'
33 OPTIMISATION = b'optimization'
34
34
35
35
36 class improvement(object):
36 class improvement(object):
37 """Represents an improvement that can be made as part of an upgrade.
37 """Represents an improvement that can be made as part of an upgrade.
38
38
39 The following attributes are defined on each instance:
39 The following attributes are defined on each instance:
40
40
41 name
41 name
42 Machine-readable string uniquely identifying this improvement. It
42 Machine-readable string uniquely identifying this improvement. It
43 will be mapped to an action later in the upgrade process.
43 will be mapped to an action later in the upgrade process.
44
44
45 type
45 type
46 Either ``FORMAT_VARIANT`` or ``OPTIMISATION``.
46 Either ``FORMAT_VARIANT`` or ``OPTIMISATION``.
47 A format variant is where we change the storage format. Not all format
47 A format variant is where we change the storage format. Not all format
48 variant changes are an obvious problem.
48 variant changes are an obvious problem.
49 An optimization is an action (sometimes optional) that
49 An optimization is an action (sometimes optional) that
50 can be taken to further improve the state of the repository.
50 can be taken to further improve the state of the repository.
51
51
52 description
52 description
53 Message intended for humans explaining the improvement in more detail,
53 Message intended for humans explaining the improvement in more detail,
54 including the implications of it. For ``FORMAT_VARIANT`` types, should be
54 including the implications of it. For ``FORMAT_VARIANT`` types, should be
55 worded in the present tense. For ``OPTIMISATION`` types, should be
55 worded in the present tense. For ``OPTIMISATION`` types, should be
56 worded in the future tense.
56 worded in the future tense.
57
57
58 upgrademessage
58 upgrademessage
59 Message intended for humans explaining what an upgrade addressing this
59 Message intended for humans explaining what an upgrade addressing this
60 issue will do. Should be worded in the future tense.
60 issue will do. Should be worded in the future tense.
61
61
62 postupgrademessage
62 postupgrademessage
63 Message intended for humans which will be shown post an upgrade
63 Message intended for humans which will be shown post an upgrade
64 operation when the improvement will be added
64 operation when the improvement will be added
65
65
66 postdowngrademessage
66 postdowngrademessage
67 Message intended for humans which will be shown post an upgrade
67 Message intended for humans which will be shown post an upgrade
68 operation in which this improvement was removed
68 operation in which this improvement was removed
69
69
70 touches_filelogs (bool)
70 touches_filelogs (bool)
71 Whether this improvement touches filelogs
71 Whether this improvement touches filelogs
72
72
73 touches_manifests (bool)
73 touches_manifests (bool)
74 Whether this improvement touches manifests
74 Whether this improvement touches manifests
75
75
76 touches_changelog (bool)
76 touches_changelog (bool)
77 Whether this improvement touches changelog
77 Whether this improvement touches changelog
78
78
79 touches_requirements (bool)
79 touches_requirements (bool)
80 Whether this improvement changes repository requirements
80 Whether this improvement changes repository requirements
81 """
81 """
82
82
83 def __init__(self, name, type, description, upgrademessage):
83 def __init__(self, name, type, description, upgrademessage):
84 self.name = name
84 self.name = name
85 self.type = type
85 self.type = type
86 self.description = description
86 self.description = description
87 self.upgrademessage = upgrademessage
87 self.upgrademessage = upgrademessage
88 self.postupgrademessage = None
88 self.postupgrademessage = None
89 self.postdowngrademessage = None
89 self.postdowngrademessage = None
90 # By default for now, we assume every improvement touches
90 # By default for now, we assume every improvement touches
91 # all the things
91 # all the things
92 self.touches_filelogs = True
92 self.touches_filelogs = True
93 self.touches_manifests = True
93 self.touches_manifests = True
94 self.touches_changelog = True
94 self.touches_changelog = True
95 self.touches_requirements = True
95 self.touches_requirements = True
96
96
97 def __eq__(self, other):
97 def __eq__(self, other):
98 if not isinstance(other, improvement):
98 if not isinstance(other, improvement):
99 # This is what python tell use to do
99 # This is what python tell use to do
100 return NotImplemented
100 return NotImplemented
101 return self.name == other.name
101 return self.name == other.name
102
102
103 def __ne__(self, other):
103 def __ne__(self, other):
104 return not (self == other)
104 return not (self == other)
105
105
106 def __hash__(self):
106 def __hash__(self):
107 return hash(self.name)
107 return hash(self.name)
108
108
109
109
110 allformatvariant = []
110 allformatvariant = []
111
111
112
112
113 def registerformatvariant(cls):
113 def registerformatvariant(cls):
114 allformatvariant.append(cls)
114 allformatvariant.append(cls)
115 return cls
115 return cls
116
116
117
117
118 class formatvariant(improvement):
118 class formatvariant(improvement):
119 """an improvement subclass dedicated to repository format"""
119 """an improvement subclass dedicated to repository format"""
120
120
121 type = FORMAT_VARIANT
121 type = FORMAT_VARIANT
122 ### The following attributes should be defined for each class:
122 ### The following attributes should be defined for each class:
123
123
124 # machine-readable string uniquely identifying this improvement. it will be
124 # machine-readable string uniquely identifying this improvement. it will be
125 # mapped to an action later in the upgrade process.
125 # mapped to an action later in the upgrade process.
126 name = None
126 name = None
127
127
128 # message intended for humans explaining the improvement in more detail,
128 # message intended for humans explaining the improvement in more detail,
129 # including the implications of it ``FORMAT_VARIANT`` types, should be
129 # including the implications of it ``FORMAT_VARIANT`` types, should be
130 # worded
130 # worded
131 # in the present tense.
131 # in the present tense.
132 description = None
132 description = None
133
133
134 # message intended for humans explaining what an upgrade addressing this
134 # message intended for humans explaining what an upgrade addressing this
135 # issue will do. should be worded in the future tense.
135 # issue will do. should be worded in the future tense.
136 upgrademessage = None
136 upgrademessage = None
137
137
138 # value of current Mercurial default for new repository
138 # value of current Mercurial default for new repository
139 default = None
139 default = None
140
140
141 # Message intended for humans which will be shown post an upgrade
141 # Message intended for humans which will be shown post an upgrade
142 # operation when the improvement will be added
142 # operation when the improvement will be added
143 postupgrademessage = None
143 postupgrademessage = None
144
144
145 # Message intended for humans which will be shown post an upgrade
145 # Message intended for humans which will be shown post an upgrade
146 # operation in which this improvement was removed
146 # operation in which this improvement was removed
147 postdowngrademessage = None
147 postdowngrademessage = None
148
148
149 # By default for now, we assume every improvement touches all the things
149 # By default for now, we assume every improvement touches all the things
150 touches_filelogs = True
150 touches_filelogs = True
151 touches_manifests = True
151 touches_manifests = True
152 touches_changelog = True
152 touches_changelog = True
153 touches_requirements = True
153 touches_requirements = True
154
154
155 def __init__(self):
155 def __init__(self):
156 raise NotImplementedError()
156 raise NotImplementedError()
157
157
158 @staticmethod
158 @staticmethod
159 def fromrepo(repo):
159 def fromrepo(repo):
160 """current value of the variant in the repository"""
160 """current value of the variant in the repository"""
161 raise NotImplementedError()
161 raise NotImplementedError()
162
162
163 @staticmethod
163 @staticmethod
164 def fromconfig(repo):
164 def fromconfig(repo):
165 """current value of the variant in the configuration"""
165 """current value of the variant in the configuration"""
166 raise NotImplementedError()
166 raise NotImplementedError()
167
167
168
168
169 class requirementformatvariant(formatvariant):
169 class requirementformatvariant(formatvariant):
170 """formatvariant based on a 'requirement' name.
170 """formatvariant based on a 'requirement' name.
171
171
172 Many format variant are controlled by a 'requirement'. We define a small
172 Many format variant are controlled by a 'requirement'. We define a small
173 subclass to factor the code.
173 subclass to factor the code.
174 """
174 """
175
175
176 # the requirement that control this format variant
176 # the requirement that control this format variant
177 _requirement = None
177 _requirement = None
178
178
179 @staticmethod
179 @staticmethod
180 def _newreporequirements(ui):
180 def _newreporequirements(ui):
181 return localrepo.newreporequirements(
181 return localrepo.newreporequirements(
182 ui, localrepo.defaultcreateopts(ui)
182 ui, localrepo.defaultcreateopts(ui)
183 )
183 )
184
184
185 @classmethod
185 @classmethod
186 def fromrepo(cls, repo):
186 def fromrepo(cls, repo):
187 assert cls._requirement is not None
187 assert cls._requirement is not None
188 return cls._requirement in repo.requirements
188 return cls._requirement in repo.requirements
189
189
190 @classmethod
190 @classmethod
191 def fromconfig(cls, repo):
191 def fromconfig(cls, repo):
192 assert cls._requirement is not None
192 assert cls._requirement is not None
193 return cls._requirement in cls._newreporequirements(repo.ui)
193 return cls._requirement in cls._newreporequirements(repo.ui)
194
194
195
195
196 @registerformatvariant
196 @registerformatvariant
197 class fncache(requirementformatvariant):
197 class fncache(requirementformatvariant):
198 name = b'fncache'
198 name = b'fncache'
199
199
200 _requirement = requirements.FNCACHE_REQUIREMENT
200 _requirement = requirements.FNCACHE_REQUIREMENT
201
201
202 default = True
202 default = True
203
203
204 description = _(
204 description = _(
205 b'long and reserved filenames may not work correctly; '
205 b'long and reserved filenames may not work correctly; '
206 b'repository performance is sub-optimal'
206 b'repository performance is sub-optimal'
207 )
207 )
208
208
209 upgrademessage = _(
209 upgrademessage = _(
210 b'repository will be more resilient to storing '
210 b'repository will be more resilient to storing '
211 b'certain paths and performance of certain '
211 b'certain paths and performance of certain '
212 b'operations should be improved'
212 b'operations should be improved'
213 )
213 )
214
214
215
215
216 @registerformatvariant
216 @registerformatvariant
217 class dotencode(requirementformatvariant):
217 class dotencode(requirementformatvariant):
218 name = b'dotencode'
218 name = b'dotencode'
219
219
220 _requirement = requirements.DOTENCODE_REQUIREMENT
220 _requirement = requirements.DOTENCODE_REQUIREMENT
221
221
222 default = True
222 default = True
223
223
224 description = _(
224 description = _(
225 b'storage of filenames beginning with a period or '
225 b'storage of filenames beginning with a period or '
226 b'space may not work correctly'
226 b'space may not work correctly'
227 )
227 )
228
228
229 upgrademessage = _(
229 upgrademessage = _(
230 b'repository will be better able to store files '
230 b'repository will be better able to store files '
231 b'beginning with a space or period'
231 b'beginning with a space or period'
232 )
232 )
233
233
234
234
235 @registerformatvariant
235 @registerformatvariant
236 class generaldelta(requirementformatvariant):
236 class generaldelta(requirementformatvariant):
237 name = b'generaldelta'
237 name = b'generaldelta'
238
238
239 _requirement = requirements.GENERALDELTA_REQUIREMENT
239 _requirement = requirements.GENERALDELTA_REQUIREMENT
240
240
241 default = True
241 default = True
242
242
243 description = _(
243 description = _(
244 b'deltas within internal storage are unable to '
244 b'deltas within internal storage are unable to '
245 b'choose optimal revisions; repository is larger and '
245 b'choose optimal revisions; repository is larger and '
246 b'slower than it could be; interaction with other '
246 b'slower than it could be; interaction with other '
247 b'repositories may require extra network and CPU '
247 b'repositories may require extra network and CPU '
248 b'resources, making "hg push" and "hg pull" slower'
248 b'resources, making "hg push" and "hg pull" slower'
249 )
249 )
250
250
251 upgrademessage = _(
251 upgrademessage = _(
252 b'repository storage will be able to create '
252 b'repository storage will be able to create '
253 b'optimal deltas; new repository data will be '
253 b'optimal deltas; new repository data will be '
254 b'smaller and read times should decrease; '
254 b'smaller and read times should decrease; '
255 b'interacting with other repositories using this '
255 b'interacting with other repositories using this '
256 b'storage model should require less network and '
256 b'storage model should require less network and '
257 b'CPU resources, making "hg push" and "hg pull" '
257 b'CPU resources, making "hg push" and "hg pull" '
258 b'faster'
258 b'faster'
259 )
259 )
260
260
261
261
262 @registerformatvariant
262 @registerformatvariant
263 class sharesafe(requirementformatvariant):
263 class sharesafe(requirementformatvariant):
264 name = b'share-safe'
264 name = b'share-safe'
265 _requirement = requirements.SHARESAFE_REQUIREMENT
265 _requirement = requirements.SHARESAFE_REQUIREMENT
266
266
267 default = False
267 default = False
268
268
269 description = _(
269 description = _(
270 b'old shared repositories do not share source repository '
270 b'old shared repositories do not share source repository '
271 b'requirements and config. This leads to various problems '
271 b'requirements and config. This leads to various problems '
272 b'when the source repository format is upgraded or some new '
272 b'when the source repository format is upgraded or some new '
273 b'extensions are enabled.'
273 b'extensions are enabled.'
274 )
274 )
275
275
276 upgrademessage = _(
276 upgrademessage = _(
277 b'Upgrades a repository to share-safe format so that future '
277 b'Upgrades a repository to share-safe format so that future '
278 b'shares of this repository share its requirements and configs.'
278 b'shares of this repository share its requirements and configs.'
279 )
279 )
280
280
281 postdowngrademessage = _(
281 postdowngrademessage = _(
282 b'repository downgraded to not use share safe mode, '
282 b'repository downgraded to not use share safe mode, '
283 b'existing shares will not work and needs to'
283 b'existing shares will not work and needs to'
284 b' be reshared.'
284 b' be reshared.'
285 )
285 )
286
286
287 postupgrademessage = _(
287 postupgrademessage = _(
288 b'repository upgraded to share safe mode, existing'
288 b'repository upgraded to share safe mode, existing'
289 b' shares will still work in old non-safe mode. '
289 b' shares will still work in old non-safe mode. '
290 b'Re-share existing shares to use them in safe mode'
290 b'Re-share existing shares to use them in safe mode'
291 b' New shares will be created in safe mode.'
291 b' New shares will be created in safe mode.'
292 )
292 )
293
293
294 # upgrade only needs to change the requirements
294 # upgrade only needs to change the requirements
295 touches_filelogs = False
295 touches_filelogs = False
296 touches_manifests = False
296 touches_manifests = False
297 touches_changelog = False
297 touches_changelog = False
298 touches_requirements = True
298 touches_requirements = True
299
299
300
300
301 @registerformatvariant
301 @registerformatvariant
302 class sparserevlog(requirementformatvariant):
302 class sparserevlog(requirementformatvariant):
303 name = b'sparserevlog'
303 name = b'sparserevlog'
304
304
305 _requirement = requirements.SPARSEREVLOG_REQUIREMENT
305 _requirement = requirements.SPARSEREVLOG_REQUIREMENT
306
306
307 default = True
307 default = True
308
308
309 description = _(
309 description = _(
310 b'in order to limit disk reading and memory usage on older '
310 b'in order to limit disk reading and memory usage on older '
311 b'version, the span of a delta chain from its root to its '
311 b'version, the span of a delta chain from its root to its '
312 b'end is limited, whatever the relevant data in this span. '
312 b'end is limited, whatever the relevant data in this span. '
313 b'This can severly limit Mercurial ability to build good '
313 b'This can severly limit Mercurial ability to build good '
314 b'chain of delta resulting is much more storage space being '
314 b'chain of delta resulting is much more storage space being '
315 b'taken and limit reusability of on disk delta during '
315 b'taken and limit reusability of on disk delta during '
316 b'exchange.'
316 b'exchange.'
317 )
317 )
318
318
319 upgrademessage = _(
319 upgrademessage = _(
320 b'Revlog supports delta chain with more unused data '
320 b'Revlog supports delta chain with more unused data '
321 b'between payload. These gaps will be skipped at read '
321 b'between payload. These gaps will be skipped at read '
322 b'time. This allows for better delta chains, making a '
322 b'time. This allows for better delta chains, making a '
323 b'better compression and faster exchange with server.'
323 b'better compression and faster exchange with server.'
324 )
324 )
325
325
326
326
327 @registerformatvariant
327 @registerformatvariant
328 class persistentnodemap(requirementformatvariant):
328 class persistentnodemap(requirementformatvariant):
329 name = b'persistent-nodemap'
329 name = b'persistent-nodemap'
330
330
331 _requirement = requirements.NODEMAP_REQUIREMENT
331 _requirement = requirements.NODEMAP_REQUIREMENT
332
332
333 default = False
333 default = False
334
334
335 description = _(
335 description = _(
336 b'persist the node -> rev mapping on disk to speedup lookup'
336 b'persist the node -> rev mapping on disk to speedup lookup'
337 )
337 )
338
338
339 upgrademessage = _(b'Speedup revision lookup by node id.')
339 upgrademessage = _(b'Speedup revision lookup by node id.')
340
340
341
341
342 @registerformatvariant
342 @registerformatvariant
343 class copiessdc(requirementformatvariant):
343 class copiessdc(requirementformatvariant):
344 name = b'copies-sdc'
344 name = b'copies-sdc'
345
345
346 _requirement = requirements.COPIESSDC_REQUIREMENT
346 _requirement = requirements.COPIESSDC_REQUIREMENT
347
347
348 default = False
348 default = False
349
349
350 description = _(b'Stores copies information alongside changesets.')
350 description = _(b'Stores copies information alongside changesets.')
351
351
352 upgrademessage = _(
352 upgrademessage = _(
353 b'Allows to use more efficient algorithm to deal with ' b'copy tracing.'
353 b'Allows to use more efficient algorithm to deal with ' b'copy tracing.'
354 )
354 )
355
355
356
356
357 @registerformatvariant
357 @registerformatvariant
358 class revlogv2(requirementformatvariant):
358 class revlogv2(requirementformatvariant):
359 name = b'revlog-v2'
359 name = b'revlog-v2'
360 _requirement = requirements.REVLOGV2_REQUIREMENT
360 _requirement = requirements.REVLOGV2_REQUIREMENT
361 default = False
361 default = False
362 description = _(b'Version 2 of the revlog.')
362 description = _(b'Version 2 of the revlog.')
363 upgrademessage = _(b'very experimental')
363 upgrademessage = _(b'very experimental')
364
364
365
365
366 @registerformatvariant
366 @registerformatvariant
367 class removecldeltachain(formatvariant):
367 class removecldeltachain(formatvariant):
368 name = b'plain-cl-delta'
368 name = b'plain-cl-delta'
369
369
370 default = True
370 default = True
371
371
372 description = _(
372 description = _(
373 b'changelog storage is using deltas instead of '
373 b'changelog storage is using deltas instead of '
374 b'raw entries; changelog reading and any '
374 b'raw entries; changelog reading and any '
375 b'operation relying on changelog data are slower '
375 b'operation relying on changelog data are slower '
376 b'than they could be'
376 b'than they could be'
377 )
377 )
378
378
379 upgrademessage = _(
379 upgrademessage = _(
380 b'changelog storage will be reformated to '
380 b'changelog storage will be reformated to '
381 b'store raw entries; changelog reading will be '
381 b'store raw entries; changelog reading will be '
382 b'faster; changelog size may be reduced'
382 b'faster; changelog size may be reduced'
383 )
383 )
384
384
385 @staticmethod
385 @staticmethod
386 def fromrepo(repo):
386 def fromrepo(repo):
387 # Mercurial 4.0 changed changelogs to not use delta chains. Search for
387 # Mercurial 4.0 changed changelogs to not use delta chains. Search for
388 # changelogs with deltas.
388 # changelogs with deltas.
389 cl = repo.changelog
389 cl = repo.changelog
390 chainbase = cl.chainbase
390 chainbase = cl.chainbase
391 return all(rev == chainbase(rev) for rev in cl)
391 return all(rev == chainbase(rev) for rev in cl)
392
392
393 @staticmethod
393 @staticmethod
394 def fromconfig(repo):
394 def fromconfig(repo):
395 return True
395 return True
396
396
397
397
398 @registerformatvariant
398 @registerformatvariant
399 class compressionengine(formatvariant):
399 class compressionengine(formatvariant):
400 name = b'compression'
400 name = b'compression'
401 default = b'zlib'
401 default = b'zlib'
402
402
403 description = _(
403 description = _(
404 b'Compresion algorithm used to compress data. '
404 b'Compresion algorithm used to compress data. '
405 b'Some engine are faster than other'
405 b'Some engine are faster than other'
406 )
406 )
407
407
408 upgrademessage = _(
408 upgrademessage = _(
409 b'revlog content will be recompressed with the new algorithm.'
409 b'revlog content will be recompressed with the new algorithm.'
410 )
410 )
411
411
412 @classmethod
412 @classmethod
413 def fromrepo(cls, repo):
413 def fromrepo(cls, repo):
414 # we allow multiple compression engine requirement to co-exist because
414 # we allow multiple compression engine requirement to co-exist because
415 # strickly speaking, revlog seems to support mixed compression style.
415 # strickly speaking, revlog seems to support mixed compression style.
416 #
416 #
417 # The compression used for new entries will be "the last one"
417 # The compression used for new entries will be "the last one"
418 compression = b'zlib'
418 compression = b'zlib'
419 for req in repo.requirements:
419 for req in repo.requirements:
420 prefix = req.startswith
420 prefix = req.startswith
421 if prefix(b'revlog-compression-') or prefix(b'exp-compression-'):
421 if prefix(b'revlog-compression-') or prefix(b'exp-compression-'):
422 compression = req.split(b'-', 2)[2]
422 compression = req.split(b'-', 2)[2]
423 return compression
423 return compression
424
424
425 @classmethod
425 @classmethod
426 def fromconfig(cls, repo):
426 def fromconfig(cls, repo):
427 compengines = repo.ui.configlist(b'format', b'revlog-compression')
427 compengines = repo.ui.configlist(b'format', b'revlog-compression')
428 # return the first valid value as the selection code would do
428 # return the first valid value as the selection code would do
429 for comp in compengines:
429 for comp in compengines:
430 if comp in util.compengines:
430 if comp in util.compengines:
431 return comp
431 e = util.compengines[comp]
432 if e.available() and e.revlogheader():
433 return comp
432
434
433 # no valide compression found lets display it all for clarity
435 # no valide compression found lets display it all for clarity
434 return b','.join(compengines)
436 return b','.join(compengines)
435
437
436
438
437 @registerformatvariant
439 @registerformatvariant
438 class compressionlevel(formatvariant):
440 class compressionlevel(formatvariant):
439 name = b'compression-level'
441 name = b'compression-level'
440 default = b'default'
442 default = b'default'
441
443
442 description = _(b'compression level')
444 description = _(b'compression level')
443
445
444 upgrademessage = _(b'revlog content will be recompressed')
446 upgrademessage = _(b'revlog content will be recompressed')
445
447
446 @classmethod
448 @classmethod
447 def fromrepo(cls, repo):
449 def fromrepo(cls, repo):
448 comp = compressionengine.fromrepo(repo)
450 comp = compressionengine.fromrepo(repo)
449 level = None
451 level = None
450 if comp == b'zlib':
452 if comp == b'zlib':
451 level = repo.ui.configint(b'storage', b'revlog.zlib.level')
453 level = repo.ui.configint(b'storage', b'revlog.zlib.level')
452 elif comp == b'zstd':
454 elif comp == b'zstd':
453 level = repo.ui.configint(b'storage', b'revlog.zstd.level')
455 level = repo.ui.configint(b'storage', b'revlog.zstd.level')
454 if level is None:
456 if level is None:
455 return b'default'
457 return b'default'
456 return bytes(level)
458 return bytes(level)
457
459
458 @classmethod
460 @classmethod
459 def fromconfig(cls, repo):
461 def fromconfig(cls, repo):
460 comp = compressionengine.fromconfig(repo)
462 comp = compressionengine.fromconfig(repo)
461 level = None
463 level = None
462 if comp == b'zlib':
464 if comp == b'zlib':
463 level = repo.ui.configint(b'storage', b'revlog.zlib.level')
465 level = repo.ui.configint(b'storage', b'revlog.zlib.level')
464 elif comp == b'zstd':
466 elif comp == b'zstd':
465 level = repo.ui.configint(b'storage', b'revlog.zstd.level')
467 level = repo.ui.configint(b'storage', b'revlog.zstd.level')
466 if level is None:
468 if level is None:
467 return b'default'
469 return b'default'
468 return bytes(level)
470 return bytes(level)
469
471
470
472
471 def find_format_upgrades(repo):
473 def find_format_upgrades(repo):
472 """returns a list of format upgrades which can be perform on the repo"""
474 """returns a list of format upgrades which can be perform on the repo"""
473 upgrades = []
475 upgrades = []
474
476
475 # We could detect lack of revlogv1 and store here, but they were added
477 # We could detect lack of revlogv1 and store here, but they were added
476 # in 0.9.2 and we don't support upgrading repos without these
478 # in 0.9.2 and we don't support upgrading repos without these
477 # requirements, so let's not bother.
479 # requirements, so let's not bother.
478
480
479 for fv in allformatvariant:
481 for fv in allformatvariant:
480 if not fv.fromrepo(repo):
482 if not fv.fromrepo(repo):
481 upgrades.append(fv)
483 upgrades.append(fv)
482
484
483 return upgrades
485 return upgrades
484
486
485
487
486 def find_format_downgrades(repo):
488 def find_format_downgrades(repo):
487 """returns a list of format downgrades which will be performed on the repo
489 """returns a list of format downgrades which will be performed on the repo
488 because of disabled config option for them"""
490 because of disabled config option for them"""
489
491
490 downgrades = []
492 downgrades = []
491
493
492 for fv in allformatvariant:
494 for fv in allformatvariant:
493 if fv.name == b'compression':
495 if fv.name == b'compression':
494 # If there is a compression change between repository
496 # If there is a compression change between repository
495 # and config, destination repository compression will change
497 # and config, destination repository compression will change
496 # and current compression will be removed.
498 # and current compression will be removed.
497 if fv.fromrepo(repo) != fv.fromconfig(repo):
499 if fv.fromrepo(repo) != fv.fromconfig(repo):
498 downgrades.append(fv)
500 downgrades.append(fv)
499 continue
501 continue
500 # format variant exist in repo but does not exist in new repository
502 # format variant exist in repo but does not exist in new repository
501 # config
503 # config
502 if fv.fromrepo(repo) and not fv.fromconfig(repo):
504 if fv.fromrepo(repo) and not fv.fromconfig(repo):
503 downgrades.append(fv)
505 downgrades.append(fv)
504
506
505 return downgrades
507 return downgrades
506
508
507
509
508 ALL_OPTIMISATIONS = []
510 ALL_OPTIMISATIONS = []
509
511
510
512
511 def register_optimization(obj):
513 def register_optimization(obj):
512 ALL_OPTIMISATIONS.append(obj)
514 ALL_OPTIMISATIONS.append(obj)
513 return obj
515 return obj
514
516
515
517
516 register_optimization(
518 register_optimization(
517 improvement(
519 improvement(
518 name=b're-delta-parent',
520 name=b're-delta-parent',
519 type=OPTIMISATION,
521 type=OPTIMISATION,
520 description=_(
522 description=_(
521 b'deltas within internal storage will be recalculated to '
523 b'deltas within internal storage will be recalculated to '
522 b'choose an optimal base revision where this was not '
524 b'choose an optimal base revision where this was not '
523 b'already done; the size of the repository may shrink and '
525 b'already done; the size of the repository may shrink and '
524 b'various operations may become faster; the first time '
526 b'various operations may become faster; the first time '
525 b'this optimization is performed could slow down upgrade '
527 b'this optimization is performed could slow down upgrade '
526 b'execution considerably; subsequent invocations should '
528 b'execution considerably; subsequent invocations should '
527 b'not run noticeably slower'
529 b'not run noticeably slower'
528 ),
530 ),
529 upgrademessage=_(
531 upgrademessage=_(
530 b'deltas within internal storage will choose a new '
532 b'deltas within internal storage will choose a new '
531 b'base revision if needed'
533 b'base revision if needed'
532 ),
534 ),
533 )
535 )
534 )
536 )
535
537
536 register_optimization(
538 register_optimization(
537 improvement(
539 improvement(
538 name=b're-delta-multibase',
540 name=b're-delta-multibase',
539 type=OPTIMISATION,
541 type=OPTIMISATION,
540 description=_(
542 description=_(
541 b'deltas within internal storage will be recalculated '
543 b'deltas within internal storage will be recalculated '
542 b'against multiple base revision and the smallest '
544 b'against multiple base revision and the smallest '
543 b'difference will be used; the size of the repository may '
545 b'difference will be used; the size of the repository may '
544 b'shrink significantly when there are many merges; this '
546 b'shrink significantly when there are many merges; this '
545 b'optimization will slow down execution in proportion to '
547 b'optimization will slow down execution in proportion to '
546 b'the number of merges in the repository and the amount '
548 b'the number of merges in the repository and the amount '
547 b'of files in the repository; this slow down should not '
549 b'of files in the repository; this slow down should not '
548 b'be significant unless there are tens of thousands of '
550 b'be significant unless there are tens of thousands of '
549 b'files and thousands of merges'
551 b'files and thousands of merges'
550 ),
552 ),
551 upgrademessage=_(
553 upgrademessage=_(
552 b'deltas within internal storage will choose an '
554 b'deltas within internal storage will choose an '
553 b'optimal delta by computing deltas against multiple '
555 b'optimal delta by computing deltas against multiple '
554 b'parents; may slow down execution time '
556 b'parents; may slow down execution time '
555 b'significantly'
557 b'significantly'
556 ),
558 ),
557 )
559 )
558 )
560 )
559
561
560 register_optimization(
562 register_optimization(
561 improvement(
563 improvement(
562 name=b're-delta-all',
564 name=b're-delta-all',
563 type=OPTIMISATION,
565 type=OPTIMISATION,
564 description=_(
566 description=_(
565 b'deltas within internal storage will always be '
567 b'deltas within internal storage will always be '
566 b'recalculated without reusing prior deltas; this will '
568 b'recalculated without reusing prior deltas; this will '
567 b'likely make execution run several times slower; this '
569 b'likely make execution run several times slower; this '
568 b'optimization is typically not needed'
570 b'optimization is typically not needed'
569 ),
571 ),
570 upgrademessage=_(
572 upgrademessage=_(
571 b'deltas within internal storage will be fully '
573 b'deltas within internal storage will be fully '
572 b'recomputed; this will likely drastically slow down '
574 b'recomputed; this will likely drastically slow down '
573 b'execution time'
575 b'execution time'
574 ),
576 ),
575 )
577 )
576 )
578 )
577
579
578 register_optimization(
580 register_optimization(
579 improvement(
581 improvement(
580 name=b're-delta-fulladd',
582 name=b're-delta-fulladd',
581 type=OPTIMISATION,
583 type=OPTIMISATION,
582 description=_(
584 description=_(
583 b'every revision will be re-added as if it was new '
585 b'every revision will be re-added as if it was new '
584 b'content. It will go through the full storage '
586 b'content. It will go through the full storage '
585 b'mechanism giving extensions a chance to process it '
587 b'mechanism giving extensions a chance to process it '
586 b'(eg. lfs). This is similar to "re-delta-all" but even '
588 b'(eg. lfs). This is similar to "re-delta-all" but even '
587 b'slower since more logic is involved.'
589 b'slower since more logic is involved.'
588 ),
590 ),
589 upgrademessage=_(
591 upgrademessage=_(
590 b'each revision will be added as new content to the '
592 b'each revision will be added as new content to the '
591 b'internal storage; this will likely drastically slow '
593 b'internal storage; this will likely drastically slow '
592 b'down execution time, but some extensions might need '
594 b'down execution time, but some extensions might need '
593 b'it'
595 b'it'
594 ),
596 ),
595 )
597 )
596 )
598 )
597
599
598
600
599 def findoptimizations(repo):
601 def findoptimizations(repo):
600 """Determine optimisation that could be used during upgrade"""
602 """Determine optimisation that could be used during upgrade"""
601 # These are unconditionally added. There is logic later that figures out
603 # These are unconditionally added. There is logic later that figures out
602 # which ones to apply.
604 # which ones to apply.
603 return list(ALL_OPTIMISATIONS)
605 return list(ALL_OPTIMISATIONS)
604
606
605
607
606 def determine_upgrade_actions(
608 def determine_upgrade_actions(
607 repo, format_upgrades, optimizations, sourcereqs, destreqs
609 repo, format_upgrades, optimizations, sourcereqs, destreqs
608 ):
610 ):
609 """Determine upgrade actions that will be performed.
611 """Determine upgrade actions that will be performed.
610
612
611 Given a list of improvements as returned by ``find_format_upgrades`` and
613 Given a list of improvements as returned by ``find_format_upgrades`` and
612 ``findoptimizations``, determine the list of upgrade actions that
614 ``findoptimizations``, determine the list of upgrade actions that
613 will be performed.
615 will be performed.
614
616
615 The role of this function is to filter improvements if needed, apply
617 The role of this function is to filter improvements if needed, apply
616 recommended optimizations from the improvements list that make sense,
618 recommended optimizations from the improvements list that make sense,
617 etc.
619 etc.
618
620
619 Returns a list of action names.
621 Returns a list of action names.
620 """
622 """
621 newactions = []
623 newactions = []
622
624
623 for d in format_upgrades:
625 for d in format_upgrades:
624 name = d._requirement
626 name = d._requirement
625
627
626 # If the action is a requirement that doesn't show up in the
628 # If the action is a requirement that doesn't show up in the
627 # destination requirements, prune the action.
629 # destination requirements, prune the action.
628 if name is not None and name not in destreqs:
630 if name is not None and name not in destreqs:
629 continue
631 continue
630
632
631 newactions.append(d)
633 newactions.append(d)
632
634
633 newactions.extend(o for o in sorted(optimizations) if o not in newactions)
635 newactions.extend(o for o in sorted(optimizations) if o not in newactions)
634
636
635 # FUTURE consider adding some optimizations here for certain transitions.
637 # FUTURE consider adding some optimizations here for certain transitions.
636 # e.g. adding generaldelta could schedule parent redeltas.
638 # e.g. adding generaldelta could schedule parent redeltas.
637
639
638 return newactions
640 return newactions
639
641
640
642
641 class UpgradeOperation(object):
643 class UpgradeOperation(object):
642 """represent the work to be done during an upgrade"""
644 """represent the work to be done during an upgrade"""
643
645
644 def __init__(
646 def __init__(
645 self,
647 self,
646 ui,
648 ui,
647 new_requirements,
649 new_requirements,
648 current_requirements,
650 current_requirements,
649 upgrade_actions,
651 upgrade_actions,
650 removed_actions,
652 removed_actions,
651 revlogs_to_process,
653 revlogs_to_process,
652 backup_store,
654 backup_store,
653 ):
655 ):
654 self.ui = ui
656 self.ui = ui
655 self.new_requirements = new_requirements
657 self.new_requirements = new_requirements
656 self.current_requirements = current_requirements
658 self.current_requirements = current_requirements
657 # list of upgrade actions the operation will perform
659 # list of upgrade actions the operation will perform
658 self.upgrade_actions = upgrade_actions
660 self.upgrade_actions = upgrade_actions
659 self._upgrade_actions_names = set([a.name for a in upgrade_actions])
661 self._upgrade_actions_names = set([a.name for a in upgrade_actions])
660 self.removed_actions = removed_actions
662 self.removed_actions = removed_actions
661 self.revlogs_to_process = revlogs_to_process
663 self.revlogs_to_process = revlogs_to_process
662 # requirements which will be added by the operation
664 # requirements which will be added by the operation
663 self._added_requirements = (
665 self._added_requirements = (
664 self.new_requirements - self.current_requirements
666 self.new_requirements - self.current_requirements
665 )
667 )
666 # requirements which will be removed by the operation
668 # requirements which will be removed by the operation
667 self._removed_requirements = (
669 self._removed_requirements = (
668 self.current_requirements - self.new_requirements
670 self.current_requirements - self.new_requirements
669 )
671 )
670 # requirements which will be preserved by the operation
672 # requirements which will be preserved by the operation
671 self._preserved_requirements = (
673 self._preserved_requirements = (
672 self.current_requirements & self.new_requirements
674 self.current_requirements & self.new_requirements
673 )
675 )
674 # optimizations which are not used and it's recommended that they
676 # optimizations which are not used and it's recommended that they
675 # should use them
677 # should use them
676 all_optimizations = findoptimizations(None)
678 all_optimizations = findoptimizations(None)
677 self.unused_optimizations = [
679 self.unused_optimizations = [
678 i for i in all_optimizations if i not in self.upgrade_actions
680 i for i in all_optimizations if i not in self.upgrade_actions
679 ]
681 ]
680
682
681 # delta reuse mode of this upgrade operation
683 # delta reuse mode of this upgrade operation
682 self.delta_reuse_mode = revlog.revlog.DELTAREUSEALWAYS
684 self.delta_reuse_mode = revlog.revlog.DELTAREUSEALWAYS
683 if b're-delta-all' in self._upgrade_actions_names:
685 if b're-delta-all' in self._upgrade_actions_names:
684 self.delta_reuse_mode = revlog.revlog.DELTAREUSENEVER
686 self.delta_reuse_mode = revlog.revlog.DELTAREUSENEVER
685 elif b're-delta-parent' in self._upgrade_actions_names:
687 elif b're-delta-parent' in self._upgrade_actions_names:
686 self.delta_reuse_mode = revlog.revlog.DELTAREUSESAMEREVS
688 self.delta_reuse_mode = revlog.revlog.DELTAREUSESAMEREVS
687 elif b're-delta-multibase' in self._upgrade_actions_names:
689 elif b're-delta-multibase' in self._upgrade_actions_names:
688 self.delta_reuse_mode = revlog.revlog.DELTAREUSESAMEREVS
690 self.delta_reuse_mode = revlog.revlog.DELTAREUSESAMEREVS
689 elif b're-delta-fulladd' in self._upgrade_actions_names:
691 elif b're-delta-fulladd' in self._upgrade_actions_names:
690 self.delta_reuse_mode = revlog.revlog.DELTAREUSEFULLADD
692 self.delta_reuse_mode = revlog.revlog.DELTAREUSEFULLADD
691
693
692 # should this operation force re-delta of both parents
694 # should this operation force re-delta of both parents
693 self.force_re_delta_both_parents = (
695 self.force_re_delta_both_parents = (
694 b're-delta-multibase' in self._upgrade_actions_names
696 b're-delta-multibase' in self._upgrade_actions_names
695 )
697 )
696
698
697 # should this operation create a backup of the store
699 # should this operation create a backup of the store
698 self.backup_store = backup_store
700 self.backup_store = backup_store
699
701
700 # whether the operation touches different revlogs at all or not
702 # whether the operation touches different revlogs at all or not
701 self.touches_filelogs = self._touches_filelogs()
703 self.touches_filelogs = self._touches_filelogs()
702 self.touches_manifests = self._touches_manifests()
704 self.touches_manifests = self._touches_manifests()
703 self.touches_changelog = self._touches_changelog()
705 self.touches_changelog = self._touches_changelog()
704 # whether the operation touches requirements file or not
706 # whether the operation touches requirements file or not
705 self.touches_requirements = self._touches_requirements()
707 self.touches_requirements = self._touches_requirements()
706 self.touches_store = (
708 self.touches_store = (
707 self.touches_filelogs
709 self.touches_filelogs
708 or self.touches_manifests
710 or self.touches_manifests
709 or self.touches_changelog
711 or self.touches_changelog
710 )
712 )
711 # does the operation only touches repository requirement
713 # does the operation only touches repository requirement
712 self.requirements_only = (
714 self.requirements_only = (
713 self.touches_requirements and not self.touches_store
715 self.touches_requirements and not self.touches_store
714 )
716 )
715
717
716 def _touches_filelogs(self):
718 def _touches_filelogs(self):
717 for a in self.upgrade_actions:
719 for a in self.upgrade_actions:
718 # in optimisations, we re-process the revlogs again
720 # in optimisations, we re-process the revlogs again
719 if a.type == OPTIMISATION:
721 if a.type == OPTIMISATION:
720 return True
722 return True
721 elif a.touches_filelogs:
723 elif a.touches_filelogs:
722 return True
724 return True
723 for a in self.removed_actions:
725 for a in self.removed_actions:
724 if a.touches_filelogs:
726 if a.touches_filelogs:
725 return True
727 return True
726 return False
728 return False
727
729
728 def _touches_manifests(self):
730 def _touches_manifests(self):
729 for a in self.upgrade_actions:
731 for a in self.upgrade_actions:
730 # in optimisations, we re-process the revlogs again
732 # in optimisations, we re-process the revlogs again
731 if a.type == OPTIMISATION:
733 if a.type == OPTIMISATION:
732 return True
734 return True
733 elif a.touches_manifests:
735 elif a.touches_manifests:
734 return True
736 return True
735 for a in self.removed_actions:
737 for a in self.removed_actions:
736 if a.touches_manifests:
738 if a.touches_manifests:
737 return True
739 return True
738 return False
740 return False
739
741
740 def _touches_changelog(self):
742 def _touches_changelog(self):
741 for a in self.upgrade_actions:
743 for a in self.upgrade_actions:
742 # in optimisations, we re-process the revlogs again
744 # in optimisations, we re-process the revlogs again
743 if a.type == OPTIMISATION:
745 if a.type == OPTIMISATION:
744 return True
746 return True
745 elif a.touches_changelog:
747 elif a.touches_changelog:
746 return True
748 return True
747 for a in self.removed_actions:
749 for a in self.removed_actions:
748 if a.touches_changelog:
750 if a.touches_changelog:
749 return True
751 return True
750 return False
752 return False
751
753
752 def _touches_requirements(self):
754 def _touches_requirements(self):
753 for a in self.upgrade_actions:
755 for a in self.upgrade_actions:
754 # optimisations are used to re-process revlogs and does not result
756 # optimisations are used to re-process revlogs and does not result
755 # in a requirement being added or removed
757 # in a requirement being added or removed
756 if a.type == OPTIMISATION:
758 if a.type == OPTIMISATION:
757 pass
759 pass
758 elif a.touches_requirements:
760 elif a.touches_requirements:
759 return True
761 return True
760 for a in self.removed_actions:
762 for a in self.removed_actions:
761 if a.touches_requirements:
763 if a.touches_requirements:
762 return True
764 return True
763
765
764 return False
766 return False
765
767
766 def _write_labeled(self, l, label):
768 def _write_labeled(self, l, label):
767 """
769 """
768 Utility function to aid writing of a list under one label
770 Utility function to aid writing of a list under one label
769 """
771 """
770 first = True
772 first = True
771 for r in sorted(l):
773 for r in sorted(l):
772 if not first:
774 if not first:
773 self.ui.write(b', ')
775 self.ui.write(b', ')
774 self.ui.write(r, label=label)
776 self.ui.write(r, label=label)
775 first = False
777 first = False
776
778
777 def print_requirements(self):
779 def print_requirements(self):
778 self.ui.write(_(b'requirements\n'))
780 self.ui.write(_(b'requirements\n'))
779 self.ui.write(_(b' preserved: '))
781 self.ui.write(_(b' preserved: '))
780 self._write_labeled(
782 self._write_labeled(
781 self._preserved_requirements, "upgrade-repo.requirement.preserved"
783 self._preserved_requirements, "upgrade-repo.requirement.preserved"
782 )
784 )
783 self.ui.write((b'\n'))
785 self.ui.write((b'\n'))
784 if self._removed_requirements:
786 if self._removed_requirements:
785 self.ui.write(_(b' removed: '))
787 self.ui.write(_(b' removed: '))
786 self._write_labeled(
788 self._write_labeled(
787 self._removed_requirements, "upgrade-repo.requirement.removed"
789 self._removed_requirements, "upgrade-repo.requirement.removed"
788 )
790 )
789 self.ui.write((b'\n'))
791 self.ui.write((b'\n'))
790 if self._added_requirements:
792 if self._added_requirements:
791 self.ui.write(_(b' added: '))
793 self.ui.write(_(b' added: '))
792 self._write_labeled(
794 self._write_labeled(
793 self._added_requirements, "upgrade-repo.requirement.added"
795 self._added_requirements, "upgrade-repo.requirement.added"
794 )
796 )
795 self.ui.write((b'\n'))
797 self.ui.write((b'\n'))
796 self.ui.write(b'\n')
798 self.ui.write(b'\n')
797
799
798 def print_optimisations(self):
800 def print_optimisations(self):
799 optimisations = [
801 optimisations = [
800 a for a in self.upgrade_actions if a.type == OPTIMISATION
802 a for a in self.upgrade_actions if a.type == OPTIMISATION
801 ]
803 ]
802 optimisations.sort(key=lambda a: a.name)
804 optimisations.sort(key=lambda a: a.name)
803 if optimisations:
805 if optimisations:
804 self.ui.write(_(b'optimisations: '))
806 self.ui.write(_(b'optimisations: '))
805 self._write_labeled(
807 self._write_labeled(
806 [a.name for a in optimisations],
808 [a.name for a in optimisations],
807 "upgrade-repo.optimisation.performed",
809 "upgrade-repo.optimisation.performed",
808 )
810 )
809 self.ui.write(b'\n\n')
811 self.ui.write(b'\n\n')
810
812
811 def print_upgrade_actions(self):
813 def print_upgrade_actions(self):
812 for a in self.upgrade_actions:
814 for a in self.upgrade_actions:
813 self.ui.status(b'%s\n %s\n\n' % (a.name, a.upgrademessage))
815 self.ui.status(b'%s\n %s\n\n' % (a.name, a.upgrademessage))
814
816
815 def print_affected_revlogs(self):
817 def print_affected_revlogs(self):
816 if not self.revlogs_to_process:
818 if not self.revlogs_to_process:
817 self.ui.write((b'no revlogs to process\n'))
819 self.ui.write((b'no revlogs to process\n'))
818 else:
820 else:
819 self.ui.write((b'processed revlogs:\n'))
821 self.ui.write((b'processed revlogs:\n'))
820 for r in sorted(self.revlogs_to_process):
822 for r in sorted(self.revlogs_to_process):
821 self.ui.write((b' - %s\n' % r))
823 self.ui.write((b' - %s\n' % r))
822 self.ui.write((b'\n'))
824 self.ui.write((b'\n'))
823
825
824 def print_unused_optimizations(self):
826 def print_unused_optimizations(self):
825 for i in self.unused_optimizations:
827 for i in self.unused_optimizations:
826 self.ui.status(_(b'%s\n %s\n\n') % (i.name, i.description))
828 self.ui.status(_(b'%s\n %s\n\n') % (i.name, i.description))
827
829
828 def has_upgrade_action(self, name):
830 def has_upgrade_action(self, name):
829 """ Check whether the upgrade operation will perform this action """
831 """ Check whether the upgrade operation will perform this action """
830 return name in self._upgrade_actions_names
832 return name in self._upgrade_actions_names
831
833
832 def print_post_op_messages(self):
834 def print_post_op_messages(self):
833 """ print post upgrade operation warning messages """
835 """ print post upgrade operation warning messages """
834 for a in self.upgrade_actions:
836 for a in self.upgrade_actions:
835 if a.postupgrademessage is not None:
837 if a.postupgrademessage is not None:
836 self.ui.warn(b'%s\n' % a.postupgrademessage)
838 self.ui.warn(b'%s\n' % a.postupgrademessage)
837 for a in self.removed_actions:
839 for a in self.removed_actions:
838 if a.postdowngrademessage is not None:
840 if a.postdowngrademessage is not None:
839 self.ui.warn(b'%s\n' % a.postdowngrademessage)
841 self.ui.warn(b'%s\n' % a.postdowngrademessage)
840
842
841
843
842 ### Code checking if a repository can got through the upgrade process at all. #
844 ### Code checking if a repository can got through the upgrade process at all. #
843
845
844
846
845 def requiredsourcerequirements(repo):
847 def requiredsourcerequirements(repo):
846 """Obtain requirements required to be present to upgrade a repo.
848 """Obtain requirements required to be present to upgrade a repo.
847
849
848 An upgrade will not be allowed if the repository doesn't have the
850 An upgrade will not be allowed if the repository doesn't have the
849 requirements returned by this function.
851 requirements returned by this function.
850 """
852 """
851 return {
853 return {
852 # Introduced in Mercurial 0.9.2.
854 # Introduced in Mercurial 0.9.2.
853 requirements.STORE_REQUIREMENT,
855 requirements.STORE_REQUIREMENT,
854 }
856 }
855
857
856
858
857 def blocksourcerequirements(repo):
859 def blocksourcerequirements(repo):
858 """Obtain requirements that will prevent an upgrade from occurring.
860 """Obtain requirements that will prevent an upgrade from occurring.
859
861
860 An upgrade cannot be performed if the source repository contains a
862 An upgrade cannot be performed if the source repository contains a
861 requirements in the returned set.
863 requirements in the returned set.
862 """
864 """
863 return {
865 return {
864 # The upgrade code does not yet support these experimental features.
866 # The upgrade code does not yet support these experimental features.
865 # This is an artificial limitation.
867 # This is an artificial limitation.
866 requirements.TREEMANIFEST_REQUIREMENT,
868 requirements.TREEMANIFEST_REQUIREMENT,
867 # This was a precursor to generaldelta and was never enabled by default.
869 # This was a precursor to generaldelta and was never enabled by default.
868 # It should (hopefully) not exist in the wild.
870 # It should (hopefully) not exist in the wild.
869 b'parentdelta',
871 b'parentdelta',
870 # Upgrade should operate on the actual store, not the shared link.
872 # Upgrade should operate on the actual store, not the shared link.
871 requirements.SHARED_REQUIREMENT,
873 requirements.SHARED_REQUIREMENT,
872 }
874 }
873
875
874
876
875 def check_revlog_version(reqs):
877 def check_revlog_version(reqs):
876 """Check that the requirements contain at least one Revlog version"""
878 """Check that the requirements contain at least one Revlog version"""
877 all_revlogs = {
879 all_revlogs = {
878 requirements.REVLOGV1_REQUIREMENT,
880 requirements.REVLOGV1_REQUIREMENT,
879 requirements.REVLOGV2_REQUIREMENT,
881 requirements.REVLOGV2_REQUIREMENT,
880 }
882 }
881 if not all_revlogs.intersection(reqs):
883 if not all_revlogs.intersection(reqs):
882 msg = _(b'cannot upgrade repository; missing a revlog version')
884 msg = _(b'cannot upgrade repository; missing a revlog version')
883 raise error.Abort(msg)
885 raise error.Abort(msg)
884
886
885
887
886 def check_source_requirements(repo):
888 def check_source_requirements(repo):
887 """Ensure that no existing requirements prevent the repository upgrade"""
889 """Ensure that no existing requirements prevent the repository upgrade"""
888
890
889 check_revlog_version(repo.requirements)
891 check_revlog_version(repo.requirements)
890 required = requiredsourcerequirements(repo)
892 required = requiredsourcerequirements(repo)
891 missingreqs = required - repo.requirements
893 missingreqs = required - repo.requirements
892 if missingreqs:
894 if missingreqs:
893 msg = _(b'cannot upgrade repository; requirement missing: %s')
895 msg = _(b'cannot upgrade repository; requirement missing: %s')
894 missingreqs = b', '.join(sorted(missingreqs))
896 missingreqs = b', '.join(sorted(missingreqs))
895 raise error.Abort(msg % missingreqs)
897 raise error.Abort(msg % missingreqs)
896
898
897 blocking = blocksourcerequirements(repo)
899 blocking = blocksourcerequirements(repo)
898 blockingreqs = blocking & repo.requirements
900 blockingreqs = blocking & repo.requirements
899 if blockingreqs:
901 if blockingreqs:
900 m = _(b'cannot upgrade repository; unsupported source requirement: %s')
902 m = _(b'cannot upgrade repository; unsupported source requirement: %s')
901 blockingreqs = b', '.join(sorted(blockingreqs))
903 blockingreqs = b', '.join(sorted(blockingreqs))
902 raise error.Abort(m % blockingreqs)
904 raise error.Abort(m % blockingreqs)
903
905
904
906
905 ### Verify the validity of the planned requirement changes ####################
907 ### Verify the validity of the planned requirement changes ####################
906
908
907
909
908 def supportremovedrequirements(repo):
910 def supportremovedrequirements(repo):
909 """Obtain requirements that can be removed during an upgrade.
911 """Obtain requirements that can be removed during an upgrade.
910
912
911 If an upgrade were to create a repository that dropped a requirement,
913 If an upgrade were to create a repository that dropped a requirement,
912 the dropped requirement must appear in the returned set for the upgrade
914 the dropped requirement must appear in the returned set for the upgrade
913 to be allowed.
915 to be allowed.
914 """
916 """
915 supported = {
917 supported = {
916 requirements.SPARSEREVLOG_REQUIREMENT,
918 requirements.SPARSEREVLOG_REQUIREMENT,
917 requirements.SIDEDATA_REQUIREMENT,
919 requirements.SIDEDATA_REQUIREMENT,
918 requirements.COPIESSDC_REQUIREMENT,
920 requirements.COPIESSDC_REQUIREMENT,
919 requirements.NODEMAP_REQUIREMENT,
921 requirements.NODEMAP_REQUIREMENT,
920 requirements.SHARESAFE_REQUIREMENT,
922 requirements.SHARESAFE_REQUIREMENT,
921 requirements.REVLOGV2_REQUIREMENT,
923 requirements.REVLOGV2_REQUIREMENT,
922 requirements.REVLOGV1_REQUIREMENT,
924 requirements.REVLOGV1_REQUIREMENT,
923 }
925 }
924 for name in compression.compengines:
926 for name in compression.compengines:
925 engine = compression.compengines[name]
927 engine = compression.compengines[name]
926 if engine.available() and engine.revlogheader():
928 if engine.available() and engine.revlogheader():
927 supported.add(b'exp-compression-%s' % name)
929 supported.add(b'exp-compression-%s' % name)
928 if engine.name() == b'zstd':
930 if engine.name() == b'zstd':
929 supported.add(b'revlog-compression-zstd')
931 supported.add(b'revlog-compression-zstd')
930 return supported
932 return supported
931
933
932
934
933 def supporteddestrequirements(repo):
935 def supporteddestrequirements(repo):
934 """Obtain requirements that upgrade supports in the destination.
936 """Obtain requirements that upgrade supports in the destination.
935
937
936 If the result of the upgrade would create requirements not in this set,
938 If the result of the upgrade would create requirements not in this set,
937 the upgrade is disallowed.
939 the upgrade is disallowed.
938
940
939 Extensions should monkeypatch this to add their custom requirements.
941 Extensions should monkeypatch this to add their custom requirements.
940 """
942 """
941 supported = {
943 supported = {
942 requirements.DOTENCODE_REQUIREMENT,
944 requirements.DOTENCODE_REQUIREMENT,
943 requirements.FNCACHE_REQUIREMENT,
945 requirements.FNCACHE_REQUIREMENT,
944 requirements.GENERALDELTA_REQUIREMENT,
946 requirements.GENERALDELTA_REQUIREMENT,
945 requirements.REVLOGV1_REQUIREMENT, # allowed in case of downgrade
947 requirements.REVLOGV1_REQUIREMENT, # allowed in case of downgrade
946 requirements.STORE_REQUIREMENT,
948 requirements.STORE_REQUIREMENT,
947 requirements.SPARSEREVLOG_REQUIREMENT,
949 requirements.SPARSEREVLOG_REQUIREMENT,
948 requirements.SIDEDATA_REQUIREMENT,
950 requirements.SIDEDATA_REQUIREMENT,
949 requirements.COPIESSDC_REQUIREMENT,
951 requirements.COPIESSDC_REQUIREMENT,
950 requirements.NODEMAP_REQUIREMENT,
952 requirements.NODEMAP_REQUIREMENT,
951 requirements.SHARESAFE_REQUIREMENT,
953 requirements.SHARESAFE_REQUIREMENT,
952 requirements.REVLOGV2_REQUIREMENT,
954 requirements.REVLOGV2_REQUIREMENT,
953 }
955 }
954 for name in compression.compengines:
956 for name in compression.compengines:
955 engine = compression.compengines[name]
957 engine = compression.compengines[name]
956 if engine.available() and engine.revlogheader():
958 if engine.available() and engine.revlogheader():
957 supported.add(b'exp-compression-%s' % name)
959 supported.add(b'exp-compression-%s' % name)
958 if engine.name() == b'zstd':
960 if engine.name() == b'zstd':
959 supported.add(b'revlog-compression-zstd')
961 supported.add(b'revlog-compression-zstd')
960 return supported
962 return supported
961
963
962
964
963 def allowednewrequirements(repo):
965 def allowednewrequirements(repo):
964 """Obtain requirements that can be added to a repository during upgrade.
966 """Obtain requirements that can be added to a repository during upgrade.
965
967
966 This is used to disallow proposed requirements from being added when
968 This is used to disallow proposed requirements from being added when
967 they weren't present before.
969 they weren't present before.
968
970
969 We use a list of allowed requirement additions instead of a list of known
971 We use a list of allowed requirement additions instead of a list of known
970 bad additions because the whitelist approach is safer and will prevent
972 bad additions because the whitelist approach is safer and will prevent
971 future, unknown requirements from accidentally being added.
973 future, unknown requirements from accidentally being added.
972 """
974 """
973 supported = {
975 supported = {
974 requirements.DOTENCODE_REQUIREMENT,
976 requirements.DOTENCODE_REQUIREMENT,
975 requirements.FNCACHE_REQUIREMENT,
977 requirements.FNCACHE_REQUIREMENT,
976 requirements.GENERALDELTA_REQUIREMENT,
978 requirements.GENERALDELTA_REQUIREMENT,
977 requirements.SPARSEREVLOG_REQUIREMENT,
979 requirements.SPARSEREVLOG_REQUIREMENT,
978 requirements.SIDEDATA_REQUIREMENT,
980 requirements.SIDEDATA_REQUIREMENT,
979 requirements.COPIESSDC_REQUIREMENT,
981 requirements.COPIESSDC_REQUIREMENT,
980 requirements.NODEMAP_REQUIREMENT,
982 requirements.NODEMAP_REQUIREMENT,
981 requirements.SHARESAFE_REQUIREMENT,
983 requirements.SHARESAFE_REQUIREMENT,
982 requirements.REVLOGV1_REQUIREMENT,
984 requirements.REVLOGV1_REQUIREMENT,
983 requirements.REVLOGV2_REQUIREMENT,
985 requirements.REVLOGV2_REQUIREMENT,
984 }
986 }
985 for name in compression.compengines:
987 for name in compression.compengines:
986 engine = compression.compengines[name]
988 engine = compression.compengines[name]
987 if engine.available() and engine.revlogheader():
989 if engine.available() and engine.revlogheader():
988 supported.add(b'exp-compression-%s' % name)
990 supported.add(b'exp-compression-%s' % name)
989 if engine.name() == b'zstd':
991 if engine.name() == b'zstd':
990 supported.add(b'revlog-compression-zstd')
992 supported.add(b'revlog-compression-zstd')
991 return supported
993 return supported
992
994
993
995
994 def check_requirements_changes(repo, new_reqs):
996 def check_requirements_changes(repo, new_reqs):
995 old_reqs = repo.requirements
997 old_reqs = repo.requirements
996 check_revlog_version(repo.requirements)
998 check_revlog_version(repo.requirements)
997 support_removal = supportremovedrequirements(repo)
999 support_removal = supportremovedrequirements(repo)
998 no_remove_reqs = old_reqs - new_reqs - support_removal
1000 no_remove_reqs = old_reqs - new_reqs - support_removal
999 if no_remove_reqs:
1001 if no_remove_reqs:
1000 msg = _(b'cannot upgrade repository; requirement would be removed: %s')
1002 msg = _(b'cannot upgrade repository; requirement would be removed: %s')
1001 no_remove_reqs = b', '.join(sorted(no_remove_reqs))
1003 no_remove_reqs = b', '.join(sorted(no_remove_reqs))
1002 raise error.Abort(msg % no_remove_reqs)
1004 raise error.Abort(msg % no_remove_reqs)
1003
1005
1004 support_addition = allowednewrequirements(repo)
1006 support_addition = allowednewrequirements(repo)
1005 no_add_reqs = new_reqs - old_reqs - support_addition
1007 no_add_reqs = new_reqs - old_reqs - support_addition
1006 if no_add_reqs:
1008 if no_add_reqs:
1007 m = _(b'cannot upgrade repository; do not support adding requirement: ')
1009 m = _(b'cannot upgrade repository; do not support adding requirement: ')
1008 no_add_reqs = b', '.join(sorted(no_add_reqs))
1010 no_add_reqs = b', '.join(sorted(no_add_reqs))
1009 raise error.Abort(m + no_add_reqs)
1011 raise error.Abort(m + no_add_reqs)
1010
1012
1011 supported = supporteddestrequirements(repo)
1013 supported = supporteddestrequirements(repo)
1012 unsupported_reqs = new_reqs - supported
1014 unsupported_reqs = new_reqs - supported
1013 if unsupported_reqs:
1015 if unsupported_reqs:
1014 msg = _(
1016 msg = _(
1015 b'cannot upgrade repository; do not support destination '
1017 b'cannot upgrade repository; do not support destination '
1016 b'requirement: %s'
1018 b'requirement: %s'
1017 )
1019 )
1018 unsupported_reqs = b', '.join(sorted(unsupported_reqs))
1020 unsupported_reqs = b', '.join(sorted(unsupported_reqs))
1019 raise error.Abort(msg % unsupported_reqs)
1021 raise error.Abort(msg % unsupported_reqs)
General Comments 0
You need to be logged in to leave comments. Login now