##// END OF EJS Templates
changectx: use names api to simplify and extend node lookup...
Sean Farley -
r23560:aead6370 default
parent child Browse files
Show More
@@ -1,1685 +1,1689 b''
1 # context.py - changeset and file context objects for mercurial
1 # context.py - changeset and file context objects for mercurial
2 #
2 #
3 # Copyright 2006, 2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 2006, 2007 Matt Mackall <mpm@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 node import nullid, nullrev, short, hex, bin
8 from node import nullid, nullrev, short, hex, bin
9 from i18n import _
9 from i18n import _
10 import mdiff, error, util, scmutil, subrepo, patch, encoding, phases
10 import mdiff, error, util, scmutil, subrepo, patch, encoding, phases
11 import match as matchmod
11 import match as matchmod
12 import os, errno, stat
12 import os, errno, stat
13 import obsolete as obsmod
13 import obsolete as obsmod
14 import repoview
14 import repoview
15 import fileset
15 import fileset
16 import revlog
16 import revlog
17
17
18 propertycache = util.propertycache
18 propertycache = util.propertycache
19
19
20 class basectx(object):
20 class basectx(object):
21 """A basectx object represents the common logic for its children:
21 """A basectx object represents the common logic for its children:
22 changectx: read-only context that is already present in the repo,
22 changectx: read-only context that is already present in the repo,
23 workingctx: a context that represents the working directory and can
23 workingctx: a context that represents the working directory and can
24 be committed,
24 be committed,
25 memctx: a context that represents changes in-memory and can also
25 memctx: a context that represents changes in-memory and can also
26 be committed."""
26 be committed."""
27 def __new__(cls, repo, changeid='', *args, **kwargs):
27 def __new__(cls, repo, changeid='', *args, **kwargs):
28 if isinstance(changeid, basectx):
28 if isinstance(changeid, basectx):
29 return changeid
29 return changeid
30
30
31 o = super(basectx, cls).__new__(cls)
31 o = super(basectx, cls).__new__(cls)
32
32
33 o._repo = repo
33 o._repo = repo
34 o._rev = nullrev
34 o._rev = nullrev
35 o._node = nullid
35 o._node = nullid
36
36
37 return o
37 return o
38
38
39 def __str__(self):
39 def __str__(self):
40 return short(self.node())
40 return short(self.node())
41
41
42 def __int__(self):
42 def __int__(self):
43 return self.rev()
43 return self.rev()
44
44
45 def __repr__(self):
45 def __repr__(self):
46 return "<%s %s>" % (type(self).__name__, str(self))
46 return "<%s %s>" % (type(self).__name__, str(self))
47
47
48 def __eq__(self, other):
48 def __eq__(self, other):
49 try:
49 try:
50 return type(self) == type(other) and self._rev == other._rev
50 return type(self) == type(other) and self._rev == other._rev
51 except AttributeError:
51 except AttributeError:
52 return False
52 return False
53
53
54 def __ne__(self, other):
54 def __ne__(self, other):
55 return not (self == other)
55 return not (self == other)
56
56
57 def __contains__(self, key):
57 def __contains__(self, key):
58 return key in self._manifest
58 return key in self._manifest
59
59
60 def __getitem__(self, key):
60 def __getitem__(self, key):
61 return self.filectx(key)
61 return self.filectx(key)
62
62
63 def __iter__(self):
63 def __iter__(self):
64 for f in sorted(self._manifest):
64 for f in sorted(self._manifest):
65 yield f
65 yield f
66
66
67 def _manifestmatches(self, match, s):
67 def _manifestmatches(self, match, s):
68 """generate a new manifest filtered by the match argument
68 """generate a new manifest filtered by the match argument
69
69
70 This method is for internal use only and mainly exists to provide an
70 This method is for internal use only and mainly exists to provide an
71 object oriented way for other contexts to customize the manifest
71 object oriented way for other contexts to customize the manifest
72 generation.
72 generation.
73 """
73 """
74 return self.manifest().matches(match)
74 return self.manifest().matches(match)
75
75
76 def _matchstatus(self, other, match):
76 def _matchstatus(self, other, match):
77 """return match.always if match is none
77 """return match.always if match is none
78
78
79 This internal method provides a way for child objects to override the
79 This internal method provides a way for child objects to override the
80 match operator.
80 match operator.
81 """
81 """
82 return match or matchmod.always(self._repo.root, self._repo.getcwd())
82 return match or matchmod.always(self._repo.root, self._repo.getcwd())
83
83
84 def _buildstatus(self, other, s, match, listignored, listclean,
84 def _buildstatus(self, other, s, match, listignored, listclean,
85 listunknown):
85 listunknown):
86 """build a status with respect to another context"""
86 """build a status with respect to another context"""
87 # Load earliest manifest first for caching reasons. More specifically,
87 # Load earliest manifest first for caching reasons. More specifically,
88 # if you have revisions 1000 and 1001, 1001 is probably stored as a
88 # if you have revisions 1000 and 1001, 1001 is probably stored as a
89 # delta against 1000. Thus, if you read 1000 first, we'll reconstruct
89 # delta against 1000. Thus, if you read 1000 first, we'll reconstruct
90 # 1000 and cache it so that when you read 1001, we just need to apply a
90 # 1000 and cache it so that when you read 1001, we just need to apply a
91 # delta to what's in the cache. So that's one full reconstruction + one
91 # delta to what's in the cache. So that's one full reconstruction + one
92 # delta application.
92 # delta application.
93 if self.rev() is not None and self.rev() < other.rev():
93 if self.rev() is not None and self.rev() < other.rev():
94 self.manifest()
94 self.manifest()
95 mf1 = other._manifestmatches(match, s)
95 mf1 = other._manifestmatches(match, s)
96 mf2 = self._manifestmatches(match, s)
96 mf2 = self._manifestmatches(match, s)
97
97
98 modified, added, clean = [], [], []
98 modified, added, clean = [], [], []
99 deleted, unknown, ignored = s.deleted, s.unknown, s.ignored
99 deleted, unknown, ignored = s.deleted, s.unknown, s.ignored
100 deletedset = set(deleted)
100 deletedset = set(deleted)
101 withflags = mf1.withflags() | mf2.withflags()
101 withflags = mf1.withflags() | mf2.withflags()
102 for fn, mf2node in mf2.iteritems():
102 for fn, mf2node in mf2.iteritems():
103 if fn in mf1:
103 if fn in mf1:
104 if (fn not in deletedset and
104 if (fn not in deletedset and
105 ((fn in withflags and mf1.flags(fn) != mf2.flags(fn)) or
105 ((fn in withflags and mf1.flags(fn) != mf2.flags(fn)) or
106 (mf1[fn] != mf2node and
106 (mf1[fn] != mf2node and
107 (mf2node or self[fn].cmp(other[fn]))))):
107 (mf2node or self[fn].cmp(other[fn]))))):
108 modified.append(fn)
108 modified.append(fn)
109 elif listclean:
109 elif listclean:
110 clean.append(fn)
110 clean.append(fn)
111 del mf1[fn]
111 del mf1[fn]
112 elif fn not in deletedset:
112 elif fn not in deletedset:
113 added.append(fn)
113 added.append(fn)
114 removed = mf1.keys()
114 removed = mf1.keys()
115 if removed:
115 if removed:
116 # need to filter files if they are already reported as removed
116 # need to filter files if they are already reported as removed
117 unknown = [fn for fn in unknown if fn not in mf1]
117 unknown = [fn for fn in unknown if fn not in mf1]
118 ignored = [fn for fn in ignored if fn not in mf1]
118 ignored = [fn for fn in ignored if fn not in mf1]
119
119
120 return scmutil.status(modified, added, removed, deleted, unknown,
120 return scmutil.status(modified, added, removed, deleted, unknown,
121 ignored, clean)
121 ignored, clean)
122
122
123 @propertycache
123 @propertycache
124 def substate(self):
124 def substate(self):
125 return subrepo.state(self, self._repo.ui)
125 return subrepo.state(self, self._repo.ui)
126
126
127 def subrev(self, subpath):
127 def subrev(self, subpath):
128 return self.substate[subpath][1]
128 return self.substate[subpath][1]
129
129
130 def rev(self):
130 def rev(self):
131 return self._rev
131 return self._rev
132 def node(self):
132 def node(self):
133 return self._node
133 return self._node
134 def hex(self):
134 def hex(self):
135 return hex(self.node())
135 return hex(self.node())
136 def manifest(self):
136 def manifest(self):
137 return self._manifest
137 return self._manifest
138 def phasestr(self):
138 def phasestr(self):
139 return phases.phasenames[self.phase()]
139 return phases.phasenames[self.phase()]
140 def mutable(self):
140 def mutable(self):
141 return self.phase() > phases.public
141 return self.phase() > phases.public
142
142
143 def getfileset(self, expr):
143 def getfileset(self, expr):
144 return fileset.getfileset(self, expr)
144 return fileset.getfileset(self, expr)
145
145
146 def obsolete(self):
146 def obsolete(self):
147 """True if the changeset is obsolete"""
147 """True if the changeset is obsolete"""
148 return self.rev() in obsmod.getrevs(self._repo, 'obsolete')
148 return self.rev() in obsmod.getrevs(self._repo, 'obsolete')
149
149
150 def extinct(self):
150 def extinct(self):
151 """True if the changeset is extinct"""
151 """True if the changeset is extinct"""
152 return self.rev() in obsmod.getrevs(self._repo, 'extinct')
152 return self.rev() in obsmod.getrevs(self._repo, 'extinct')
153
153
154 def unstable(self):
154 def unstable(self):
155 """True if the changeset is not obsolete but it's ancestor are"""
155 """True if the changeset is not obsolete but it's ancestor are"""
156 return self.rev() in obsmod.getrevs(self._repo, 'unstable')
156 return self.rev() in obsmod.getrevs(self._repo, 'unstable')
157
157
158 def bumped(self):
158 def bumped(self):
159 """True if the changeset try to be a successor of a public changeset
159 """True if the changeset try to be a successor of a public changeset
160
160
161 Only non-public and non-obsolete changesets may be bumped.
161 Only non-public and non-obsolete changesets may be bumped.
162 """
162 """
163 return self.rev() in obsmod.getrevs(self._repo, 'bumped')
163 return self.rev() in obsmod.getrevs(self._repo, 'bumped')
164
164
165 def divergent(self):
165 def divergent(self):
166 """Is a successors of a changeset with multiple possible successors set
166 """Is a successors of a changeset with multiple possible successors set
167
167
168 Only non-public and non-obsolete changesets may be divergent.
168 Only non-public and non-obsolete changesets may be divergent.
169 """
169 """
170 return self.rev() in obsmod.getrevs(self._repo, 'divergent')
170 return self.rev() in obsmod.getrevs(self._repo, 'divergent')
171
171
172 def troubled(self):
172 def troubled(self):
173 """True if the changeset is either unstable, bumped or divergent"""
173 """True if the changeset is either unstable, bumped or divergent"""
174 return self.unstable() or self.bumped() or self.divergent()
174 return self.unstable() or self.bumped() or self.divergent()
175
175
176 def troubles(self):
176 def troubles(self):
177 """return the list of troubles affecting this changesets.
177 """return the list of troubles affecting this changesets.
178
178
179 Troubles are returned as strings. possible values are:
179 Troubles are returned as strings. possible values are:
180 - unstable,
180 - unstable,
181 - bumped,
181 - bumped,
182 - divergent.
182 - divergent.
183 """
183 """
184 troubles = []
184 troubles = []
185 if self.unstable():
185 if self.unstable():
186 troubles.append('unstable')
186 troubles.append('unstable')
187 if self.bumped():
187 if self.bumped():
188 troubles.append('bumped')
188 troubles.append('bumped')
189 if self.divergent():
189 if self.divergent():
190 troubles.append('divergent')
190 troubles.append('divergent')
191 return troubles
191 return troubles
192
192
193 def parents(self):
193 def parents(self):
194 """return contexts for each parent changeset"""
194 """return contexts for each parent changeset"""
195 return self._parents
195 return self._parents
196
196
197 def p1(self):
197 def p1(self):
198 return self._parents[0]
198 return self._parents[0]
199
199
200 def p2(self):
200 def p2(self):
201 if len(self._parents) == 2:
201 if len(self._parents) == 2:
202 return self._parents[1]
202 return self._parents[1]
203 return changectx(self._repo, -1)
203 return changectx(self._repo, -1)
204
204
205 def _fileinfo(self, path):
205 def _fileinfo(self, path):
206 if '_manifest' in self.__dict__:
206 if '_manifest' in self.__dict__:
207 try:
207 try:
208 return self._manifest[path], self._manifest.flags(path)
208 return self._manifest[path], self._manifest.flags(path)
209 except KeyError:
209 except KeyError:
210 raise error.ManifestLookupError(self._node, path,
210 raise error.ManifestLookupError(self._node, path,
211 _('not found in manifest'))
211 _('not found in manifest'))
212 if '_manifestdelta' in self.__dict__ or path in self.files():
212 if '_manifestdelta' in self.__dict__ or path in self.files():
213 if path in self._manifestdelta:
213 if path in self._manifestdelta:
214 return (self._manifestdelta[path],
214 return (self._manifestdelta[path],
215 self._manifestdelta.flags(path))
215 self._manifestdelta.flags(path))
216 node, flag = self._repo.manifest.find(self._changeset[0], path)
216 node, flag = self._repo.manifest.find(self._changeset[0], path)
217 if not node:
217 if not node:
218 raise error.ManifestLookupError(self._node, path,
218 raise error.ManifestLookupError(self._node, path,
219 _('not found in manifest'))
219 _('not found in manifest'))
220
220
221 return node, flag
221 return node, flag
222
222
223 def filenode(self, path):
223 def filenode(self, path):
224 return self._fileinfo(path)[0]
224 return self._fileinfo(path)[0]
225
225
226 def flags(self, path):
226 def flags(self, path):
227 try:
227 try:
228 return self._fileinfo(path)[1]
228 return self._fileinfo(path)[1]
229 except error.LookupError:
229 except error.LookupError:
230 return ''
230 return ''
231
231
232 def sub(self, path):
232 def sub(self, path):
233 return subrepo.subrepo(self, path)
233 return subrepo.subrepo(self, path)
234
234
235 def match(self, pats=[], include=None, exclude=None, default='glob'):
235 def match(self, pats=[], include=None, exclude=None, default='glob'):
236 r = self._repo
236 r = self._repo
237 return matchmod.match(r.root, r.getcwd(), pats,
237 return matchmod.match(r.root, r.getcwd(), pats,
238 include, exclude, default,
238 include, exclude, default,
239 auditor=r.auditor, ctx=self)
239 auditor=r.auditor, ctx=self)
240
240
241 def diff(self, ctx2=None, match=None, **opts):
241 def diff(self, ctx2=None, match=None, **opts):
242 """Returns a diff generator for the given contexts and matcher"""
242 """Returns a diff generator for the given contexts and matcher"""
243 if ctx2 is None:
243 if ctx2 is None:
244 ctx2 = self.p1()
244 ctx2 = self.p1()
245 if ctx2 is not None:
245 if ctx2 is not None:
246 ctx2 = self._repo[ctx2]
246 ctx2 = self._repo[ctx2]
247 diffopts = patch.diffopts(self._repo.ui, opts)
247 diffopts = patch.diffopts(self._repo.ui, opts)
248 return patch.diff(self._repo, ctx2, self, match=match, opts=diffopts)
248 return patch.diff(self._repo, ctx2, self, match=match, opts=diffopts)
249
249
250 @propertycache
250 @propertycache
251 def _dirs(self):
251 def _dirs(self):
252 return scmutil.dirs(self._manifest)
252 return scmutil.dirs(self._manifest)
253
253
254 def dirs(self):
254 def dirs(self):
255 return self._dirs
255 return self._dirs
256
256
257 def dirty(self, missing=False, merge=True, branch=True):
257 def dirty(self, missing=False, merge=True, branch=True):
258 return False
258 return False
259
259
260 def status(self, other=None, match=None, listignored=False,
260 def status(self, other=None, match=None, listignored=False,
261 listclean=False, listunknown=False, listsubrepos=False):
261 listclean=False, listunknown=False, listsubrepos=False):
262 """return status of files between two nodes or node and working
262 """return status of files between two nodes or node and working
263 directory.
263 directory.
264
264
265 If other is None, compare this node with working directory.
265 If other is None, compare this node with working directory.
266
266
267 returns (modified, added, removed, deleted, unknown, ignored, clean)
267 returns (modified, added, removed, deleted, unknown, ignored, clean)
268 """
268 """
269
269
270 ctx1 = self
270 ctx1 = self
271 ctx2 = self._repo[other]
271 ctx2 = self._repo[other]
272
272
273 # This next code block is, admittedly, fragile logic that tests for
273 # This next code block is, admittedly, fragile logic that tests for
274 # reversing the contexts and wouldn't need to exist if it weren't for
274 # reversing the contexts and wouldn't need to exist if it weren't for
275 # the fast (and common) code path of comparing the working directory
275 # the fast (and common) code path of comparing the working directory
276 # with its first parent.
276 # with its first parent.
277 #
277 #
278 # What we're aiming for here is the ability to call:
278 # What we're aiming for here is the ability to call:
279 #
279 #
280 # workingctx.status(parentctx)
280 # workingctx.status(parentctx)
281 #
281 #
282 # If we always built the manifest for each context and compared those,
282 # If we always built the manifest for each context and compared those,
283 # then we'd be done. But the special case of the above call means we
283 # then we'd be done. But the special case of the above call means we
284 # just copy the manifest of the parent.
284 # just copy the manifest of the parent.
285 reversed = False
285 reversed = False
286 if (not isinstance(ctx1, changectx)
286 if (not isinstance(ctx1, changectx)
287 and isinstance(ctx2, changectx)):
287 and isinstance(ctx2, changectx)):
288 reversed = True
288 reversed = True
289 ctx1, ctx2 = ctx2, ctx1
289 ctx1, ctx2 = ctx2, ctx1
290
290
291 match = ctx2._matchstatus(ctx1, match)
291 match = ctx2._matchstatus(ctx1, match)
292 r = scmutil.status([], [], [], [], [], [], [])
292 r = scmutil.status([], [], [], [], [], [], [])
293 r = ctx2._buildstatus(ctx1, r, match, listignored, listclean,
293 r = ctx2._buildstatus(ctx1, r, match, listignored, listclean,
294 listunknown)
294 listunknown)
295
295
296 if reversed:
296 if reversed:
297 # Reverse added and removed. Clear deleted, unknown and ignored as
297 # Reverse added and removed. Clear deleted, unknown and ignored as
298 # these make no sense to reverse.
298 # these make no sense to reverse.
299 r = scmutil.status(r.modified, r.removed, r.added, [], [], [],
299 r = scmutil.status(r.modified, r.removed, r.added, [], [], [],
300 r.clean)
300 r.clean)
301
301
302 if listsubrepos:
302 if listsubrepos:
303 for subpath, sub in scmutil.itersubrepos(ctx1, ctx2):
303 for subpath, sub in scmutil.itersubrepos(ctx1, ctx2):
304 rev2 = ctx2.subrev(subpath)
304 rev2 = ctx2.subrev(subpath)
305 try:
305 try:
306 submatch = matchmod.narrowmatcher(subpath, match)
306 submatch = matchmod.narrowmatcher(subpath, match)
307 s = sub.status(rev2, match=submatch, ignored=listignored,
307 s = sub.status(rev2, match=submatch, ignored=listignored,
308 clean=listclean, unknown=listunknown,
308 clean=listclean, unknown=listunknown,
309 listsubrepos=True)
309 listsubrepos=True)
310 for rfiles, sfiles in zip(r, s):
310 for rfiles, sfiles in zip(r, s):
311 rfiles.extend("%s/%s" % (subpath, f) for f in sfiles)
311 rfiles.extend("%s/%s" % (subpath, f) for f in sfiles)
312 except error.LookupError:
312 except error.LookupError:
313 self._repo.ui.status(_("skipping missing "
313 self._repo.ui.status(_("skipping missing "
314 "subrepository: %s\n") % subpath)
314 "subrepository: %s\n") % subpath)
315
315
316 for l in r:
316 for l in r:
317 l.sort()
317 l.sort()
318
318
319 return r
319 return r
320
320
321
321
322 def makememctx(repo, parents, text, user, date, branch, files, store,
322 def makememctx(repo, parents, text, user, date, branch, files, store,
323 editor=None):
323 editor=None):
324 def getfilectx(repo, memctx, path):
324 def getfilectx(repo, memctx, path):
325 data, mode, copied = store.getfile(path)
325 data, mode, copied = store.getfile(path)
326 if data is None:
326 if data is None:
327 return None
327 return None
328 islink, isexec = mode
328 islink, isexec = mode
329 return memfilectx(repo, path, data, islink=islink, isexec=isexec,
329 return memfilectx(repo, path, data, islink=islink, isexec=isexec,
330 copied=copied, memctx=memctx)
330 copied=copied, memctx=memctx)
331 extra = {}
331 extra = {}
332 if branch:
332 if branch:
333 extra['branch'] = encoding.fromlocal(branch)
333 extra['branch'] = encoding.fromlocal(branch)
334 ctx = memctx(repo, parents, text, files, getfilectx, user,
334 ctx = memctx(repo, parents, text, files, getfilectx, user,
335 date, extra, editor)
335 date, extra, editor)
336 return ctx
336 return ctx
337
337
338 class changectx(basectx):
338 class changectx(basectx):
339 """A changecontext object makes access to data related to a particular
339 """A changecontext object makes access to data related to a particular
340 changeset convenient. It represents a read-only context already present in
340 changeset convenient. It represents a read-only context already present in
341 the repo."""
341 the repo."""
342 def __init__(self, repo, changeid=''):
342 def __init__(self, repo, changeid=''):
343 """changeid is a revision number, node, or tag"""
343 """changeid is a revision number, node, or tag"""
344
344
345 # since basectx.__new__ already took care of copying the object, we
345 # since basectx.__new__ already took care of copying the object, we
346 # don't need to do anything in __init__, so we just exit here
346 # don't need to do anything in __init__, so we just exit here
347 if isinstance(changeid, basectx):
347 if isinstance(changeid, basectx):
348 return
348 return
349
349
350 if changeid == '':
350 if changeid == '':
351 changeid = '.'
351 changeid = '.'
352 self._repo = repo
352 self._repo = repo
353
353
354 try:
354 try:
355 if isinstance(changeid, int):
355 if isinstance(changeid, int):
356 self._node = repo.changelog.node(changeid)
356 self._node = repo.changelog.node(changeid)
357 self._rev = changeid
357 self._rev = changeid
358 return
358 return
359 if isinstance(changeid, long):
359 if isinstance(changeid, long):
360 changeid = str(changeid)
360 changeid = str(changeid)
361 if changeid == '.':
361 if changeid == '.':
362 self._node = repo.dirstate.p1()
362 self._node = repo.dirstate.p1()
363 self._rev = repo.changelog.rev(self._node)
363 self._rev = repo.changelog.rev(self._node)
364 return
364 return
365 if changeid == 'null':
365 if changeid == 'null':
366 self._node = nullid
366 self._node = nullid
367 self._rev = nullrev
367 self._rev = nullrev
368 return
368 return
369 if changeid == 'tip':
369 if changeid == 'tip':
370 self._node = repo.changelog.tip()
370 self._node = repo.changelog.tip()
371 self._rev = repo.changelog.rev(self._node)
371 self._rev = repo.changelog.rev(self._node)
372 return
372 return
373 if len(changeid) == 20:
373 if len(changeid) == 20:
374 try:
374 try:
375 self._node = changeid
375 self._node = changeid
376 self._rev = repo.changelog.rev(changeid)
376 self._rev = repo.changelog.rev(changeid)
377 return
377 return
378 except error.FilteredRepoLookupError:
378 except error.FilteredRepoLookupError:
379 raise
379 raise
380 except LookupError:
380 except LookupError:
381 pass
381 pass
382
382
383 try:
383 try:
384 r = int(changeid)
384 r = int(changeid)
385 if str(r) != changeid:
385 if str(r) != changeid:
386 raise ValueError
386 raise ValueError
387 l = len(repo.changelog)
387 l = len(repo.changelog)
388 if r < 0:
388 if r < 0:
389 r += l
389 r += l
390 if r < 0 or r >= l:
390 if r < 0 or r >= l:
391 raise ValueError
391 raise ValueError
392 self._rev = r
392 self._rev = r
393 self._node = repo.changelog.node(r)
393 self._node = repo.changelog.node(r)
394 return
394 return
395 except error.FilteredIndexError:
395 except error.FilteredIndexError:
396 raise
396 raise
397 except (ValueError, OverflowError, IndexError):
397 except (ValueError, OverflowError, IndexError):
398 pass
398 pass
399
399
400 if len(changeid) == 40:
400 if len(changeid) == 40:
401 try:
401 try:
402 self._node = bin(changeid)
402 self._node = bin(changeid)
403 self._rev = repo.changelog.rev(self._node)
403 self._rev = repo.changelog.rev(self._node)
404 return
404 return
405 except error.FilteredLookupError:
405 except error.FilteredLookupError:
406 raise
406 raise
407 except (TypeError, LookupError):
407 except (TypeError, LookupError):
408 pass
408 pass
409
409
410 if changeid in repo._bookmarks:
410 # lookup bookmarks through the name interface
411 self._node = repo._bookmarks[changeid]
411 try:
412 self._node = repo.names.singlenode(changeid)
412 self._rev = repo.changelog.rev(self._node)
413 self._rev = repo.changelog.rev(self._node)
413 return
414 return
415 except KeyError:
416 pass
417
414 if changeid in repo._tagscache.tags:
418 if changeid in repo._tagscache.tags:
415 self._node = repo._tagscache.tags[changeid]
419 self._node = repo._tagscache.tags[changeid]
416 self._rev = repo.changelog.rev(self._node)
420 self._rev = repo.changelog.rev(self._node)
417 return
421 return
418 try:
422 try:
419 self._node = repo.branchtip(changeid)
423 self._node = repo.branchtip(changeid)
420 self._rev = repo.changelog.rev(self._node)
424 self._rev = repo.changelog.rev(self._node)
421 return
425 return
422 except error.FilteredRepoLookupError:
426 except error.FilteredRepoLookupError:
423 raise
427 raise
424 except error.RepoLookupError:
428 except error.RepoLookupError:
425 pass
429 pass
426
430
427 self._node = repo.unfiltered().changelog._partialmatch(changeid)
431 self._node = repo.unfiltered().changelog._partialmatch(changeid)
428 if self._node is not None:
432 if self._node is not None:
429 self._rev = repo.changelog.rev(self._node)
433 self._rev = repo.changelog.rev(self._node)
430 return
434 return
431
435
432 # lookup failed
436 # lookup failed
433 # check if it might have come from damaged dirstate
437 # check if it might have come from damaged dirstate
434 #
438 #
435 # XXX we could avoid the unfiltered if we had a recognizable
439 # XXX we could avoid the unfiltered if we had a recognizable
436 # exception for filtered changeset access
440 # exception for filtered changeset access
437 if changeid in repo.unfiltered().dirstate.parents():
441 if changeid in repo.unfiltered().dirstate.parents():
438 msg = _("working directory has unknown parent '%s'!")
442 msg = _("working directory has unknown parent '%s'!")
439 raise error.Abort(msg % short(changeid))
443 raise error.Abort(msg % short(changeid))
440 try:
444 try:
441 if len(changeid) == 20:
445 if len(changeid) == 20:
442 changeid = hex(changeid)
446 changeid = hex(changeid)
443 except TypeError:
447 except TypeError:
444 pass
448 pass
445 except (error.FilteredIndexError, error.FilteredLookupError,
449 except (error.FilteredIndexError, error.FilteredLookupError,
446 error.FilteredRepoLookupError):
450 error.FilteredRepoLookupError):
447 if repo.filtername == 'visible':
451 if repo.filtername == 'visible':
448 msg = _("hidden revision '%s'") % changeid
452 msg = _("hidden revision '%s'") % changeid
449 hint = _('use --hidden to access hidden revisions')
453 hint = _('use --hidden to access hidden revisions')
450 raise error.FilteredRepoLookupError(msg, hint=hint)
454 raise error.FilteredRepoLookupError(msg, hint=hint)
451 msg = _("filtered revision '%s' (not in '%s' subset)")
455 msg = _("filtered revision '%s' (not in '%s' subset)")
452 msg %= (changeid, repo.filtername)
456 msg %= (changeid, repo.filtername)
453 raise error.FilteredRepoLookupError(msg)
457 raise error.FilteredRepoLookupError(msg)
454 except IndexError:
458 except IndexError:
455 pass
459 pass
456 raise error.RepoLookupError(
460 raise error.RepoLookupError(
457 _("unknown revision '%s'") % changeid)
461 _("unknown revision '%s'") % changeid)
458
462
459 def __hash__(self):
463 def __hash__(self):
460 try:
464 try:
461 return hash(self._rev)
465 return hash(self._rev)
462 except AttributeError:
466 except AttributeError:
463 return id(self)
467 return id(self)
464
468
465 def __nonzero__(self):
469 def __nonzero__(self):
466 return self._rev != nullrev
470 return self._rev != nullrev
467
471
468 @propertycache
472 @propertycache
469 def _changeset(self):
473 def _changeset(self):
470 return self._repo.changelog.read(self.rev())
474 return self._repo.changelog.read(self.rev())
471
475
472 @propertycache
476 @propertycache
473 def _manifest(self):
477 def _manifest(self):
474 return self._repo.manifest.read(self._changeset[0])
478 return self._repo.manifest.read(self._changeset[0])
475
479
476 @propertycache
480 @propertycache
477 def _manifestdelta(self):
481 def _manifestdelta(self):
478 return self._repo.manifest.readdelta(self._changeset[0])
482 return self._repo.manifest.readdelta(self._changeset[0])
479
483
480 @propertycache
484 @propertycache
481 def _parents(self):
485 def _parents(self):
482 p = self._repo.changelog.parentrevs(self._rev)
486 p = self._repo.changelog.parentrevs(self._rev)
483 if p[1] == nullrev:
487 if p[1] == nullrev:
484 p = p[:-1]
488 p = p[:-1]
485 return [changectx(self._repo, x) for x in p]
489 return [changectx(self._repo, x) for x in p]
486
490
487 def changeset(self):
491 def changeset(self):
488 return self._changeset
492 return self._changeset
489 def manifestnode(self):
493 def manifestnode(self):
490 return self._changeset[0]
494 return self._changeset[0]
491
495
492 def user(self):
496 def user(self):
493 return self._changeset[1]
497 return self._changeset[1]
494 def date(self):
498 def date(self):
495 return self._changeset[2]
499 return self._changeset[2]
496 def files(self):
500 def files(self):
497 return self._changeset[3]
501 return self._changeset[3]
498 def description(self):
502 def description(self):
499 return self._changeset[4]
503 return self._changeset[4]
500 def branch(self):
504 def branch(self):
501 return encoding.tolocal(self._changeset[5].get("branch"))
505 return encoding.tolocal(self._changeset[5].get("branch"))
502 def closesbranch(self):
506 def closesbranch(self):
503 return 'close' in self._changeset[5]
507 return 'close' in self._changeset[5]
504 def extra(self):
508 def extra(self):
505 return self._changeset[5]
509 return self._changeset[5]
506 def tags(self):
510 def tags(self):
507 return self._repo.nodetags(self._node)
511 return self._repo.nodetags(self._node)
508 def bookmarks(self):
512 def bookmarks(self):
509 return self._repo.nodebookmarks(self._node)
513 return self._repo.nodebookmarks(self._node)
510 def phase(self):
514 def phase(self):
511 return self._repo._phasecache.phase(self._repo, self._rev)
515 return self._repo._phasecache.phase(self._repo, self._rev)
512 def hidden(self):
516 def hidden(self):
513 return self._rev in repoview.filterrevs(self._repo, 'visible')
517 return self._rev in repoview.filterrevs(self._repo, 'visible')
514
518
515 def children(self):
519 def children(self):
516 """return contexts for each child changeset"""
520 """return contexts for each child changeset"""
517 c = self._repo.changelog.children(self._node)
521 c = self._repo.changelog.children(self._node)
518 return [changectx(self._repo, x) for x in c]
522 return [changectx(self._repo, x) for x in c]
519
523
520 def ancestors(self):
524 def ancestors(self):
521 for a in self._repo.changelog.ancestors([self._rev]):
525 for a in self._repo.changelog.ancestors([self._rev]):
522 yield changectx(self._repo, a)
526 yield changectx(self._repo, a)
523
527
524 def descendants(self):
528 def descendants(self):
525 for d in self._repo.changelog.descendants([self._rev]):
529 for d in self._repo.changelog.descendants([self._rev]):
526 yield changectx(self._repo, d)
530 yield changectx(self._repo, d)
527
531
528 def filectx(self, path, fileid=None, filelog=None):
532 def filectx(self, path, fileid=None, filelog=None):
529 """get a file context from this changeset"""
533 """get a file context from this changeset"""
530 if fileid is None:
534 if fileid is None:
531 fileid = self.filenode(path)
535 fileid = self.filenode(path)
532 return filectx(self._repo, path, fileid=fileid,
536 return filectx(self._repo, path, fileid=fileid,
533 changectx=self, filelog=filelog)
537 changectx=self, filelog=filelog)
534
538
535 def ancestor(self, c2, warn=False):
539 def ancestor(self, c2, warn=False):
536 """return the "best" ancestor context of self and c2
540 """return the "best" ancestor context of self and c2
537
541
538 If there are multiple candidates, it will show a message and check
542 If there are multiple candidates, it will show a message and check
539 merge.preferancestor configuration before falling back to the
543 merge.preferancestor configuration before falling back to the
540 revlog ancestor."""
544 revlog ancestor."""
541 # deal with workingctxs
545 # deal with workingctxs
542 n2 = c2._node
546 n2 = c2._node
543 if n2 is None:
547 if n2 is None:
544 n2 = c2._parents[0]._node
548 n2 = c2._parents[0]._node
545 cahs = self._repo.changelog.commonancestorsheads(self._node, n2)
549 cahs = self._repo.changelog.commonancestorsheads(self._node, n2)
546 if not cahs:
550 if not cahs:
547 anc = nullid
551 anc = nullid
548 elif len(cahs) == 1:
552 elif len(cahs) == 1:
549 anc = cahs[0]
553 anc = cahs[0]
550 else:
554 else:
551 for r in self._repo.ui.configlist('merge', 'preferancestor'):
555 for r in self._repo.ui.configlist('merge', 'preferancestor'):
552 try:
556 try:
553 ctx = changectx(self._repo, r)
557 ctx = changectx(self._repo, r)
554 except error.RepoLookupError:
558 except error.RepoLookupError:
555 continue
559 continue
556 anc = ctx.node()
560 anc = ctx.node()
557 if anc in cahs:
561 if anc in cahs:
558 break
562 break
559 else:
563 else:
560 anc = self._repo.changelog.ancestor(self._node, n2)
564 anc = self._repo.changelog.ancestor(self._node, n2)
561 if warn:
565 if warn:
562 self._repo.ui.status(
566 self._repo.ui.status(
563 (_("note: using %s as ancestor of %s and %s\n") %
567 (_("note: using %s as ancestor of %s and %s\n") %
564 (short(anc), short(self._node), short(n2))) +
568 (short(anc), short(self._node), short(n2))) +
565 ''.join(_(" alternatively, use --config "
569 ''.join(_(" alternatively, use --config "
566 "merge.preferancestor=%s\n") %
570 "merge.preferancestor=%s\n") %
567 short(n) for n in sorted(cahs) if n != anc))
571 short(n) for n in sorted(cahs) if n != anc))
568 return changectx(self._repo, anc)
572 return changectx(self._repo, anc)
569
573
570 def descendant(self, other):
574 def descendant(self, other):
571 """True if other is descendant of this changeset"""
575 """True if other is descendant of this changeset"""
572 return self._repo.changelog.descendant(self._rev, other._rev)
576 return self._repo.changelog.descendant(self._rev, other._rev)
573
577
574 def walk(self, match):
578 def walk(self, match):
575 fset = set(match.files())
579 fset = set(match.files())
576 # for dirstate.walk, files=['.'] means "walk the whole tree".
580 # for dirstate.walk, files=['.'] means "walk the whole tree".
577 # follow that here, too
581 # follow that here, too
578 fset.discard('.')
582 fset.discard('.')
579
583
580 # avoid the entire walk if we're only looking for specific files
584 # avoid the entire walk if we're only looking for specific files
581 if fset and not match.anypats():
585 if fset and not match.anypats():
582 if util.all([fn in self for fn in fset]):
586 if util.all([fn in self for fn in fset]):
583 for fn in sorted(fset):
587 for fn in sorted(fset):
584 if match(fn):
588 if match(fn):
585 yield fn
589 yield fn
586 raise StopIteration
590 raise StopIteration
587
591
588 for fn in self:
592 for fn in self:
589 if fn in fset:
593 if fn in fset:
590 # specified pattern is the exact name
594 # specified pattern is the exact name
591 fset.remove(fn)
595 fset.remove(fn)
592 if match(fn):
596 if match(fn):
593 yield fn
597 yield fn
594 for fn in sorted(fset):
598 for fn in sorted(fset):
595 if fn in self._dirs:
599 if fn in self._dirs:
596 # specified pattern is a directory
600 # specified pattern is a directory
597 continue
601 continue
598 match.bad(fn, _('no such file in rev %s') % self)
602 match.bad(fn, _('no such file in rev %s') % self)
599
603
600 def matches(self, match):
604 def matches(self, match):
601 return self.walk(match)
605 return self.walk(match)
602
606
603 class basefilectx(object):
607 class basefilectx(object):
604 """A filecontext object represents the common logic for its children:
608 """A filecontext object represents the common logic for its children:
605 filectx: read-only access to a filerevision that is already present
609 filectx: read-only access to a filerevision that is already present
606 in the repo,
610 in the repo,
607 workingfilectx: a filecontext that represents files from the working
611 workingfilectx: a filecontext that represents files from the working
608 directory,
612 directory,
609 memfilectx: a filecontext that represents files in-memory."""
613 memfilectx: a filecontext that represents files in-memory."""
610 def __new__(cls, repo, path, *args, **kwargs):
614 def __new__(cls, repo, path, *args, **kwargs):
611 return super(basefilectx, cls).__new__(cls)
615 return super(basefilectx, cls).__new__(cls)
612
616
613 @propertycache
617 @propertycache
614 def _filelog(self):
618 def _filelog(self):
615 return self._repo.file(self._path)
619 return self._repo.file(self._path)
616
620
617 @propertycache
621 @propertycache
618 def _changeid(self):
622 def _changeid(self):
619 if '_changeid' in self.__dict__:
623 if '_changeid' in self.__dict__:
620 return self._changeid
624 return self._changeid
621 elif '_changectx' in self.__dict__:
625 elif '_changectx' in self.__dict__:
622 return self._changectx.rev()
626 return self._changectx.rev()
623 else:
627 else:
624 return self._filelog.linkrev(self._filerev)
628 return self._filelog.linkrev(self._filerev)
625
629
626 @propertycache
630 @propertycache
627 def _filenode(self):
631 def _filenode(self):
628 if '_fileid' in self.__dict__:
632 if '_fileid' in self.__dict__:
629 return self._filelog.lookup(self._fileid)
633 return self._filelog.lookup(self._fileid)
630 else:
634 else:
631 return self._changectx.filenode(self._path)
635 return self._changectx.filenode(self._path)
632
636
633 @propertycache
637 @propertycache
634 def _filerev(self):
638 def _filerev(self):
635 return self._filelog.rev(self._filenode)
639 return self._filelog.rev(self._filenode)
636
640
637 @propertycache
641 @propertycache
638 def _repopath(self):
642 def _repopath(self):
639 return self._path
643 return self._path
640
644
641 def __nonzero__(self):
645 def __nonzero__(self):
642 try:
646 try:
643 self._filenode
647 self._filenode
644 return True
648 return True
645 except error.LookupError:
649 except error.LookupError:
646 # file is missing
650 # file is missing
647 return False
651 return False
648
652
649 def __str__(self):
653 def __str__(self):
650 return "%s@%s" % (self.path(), self._changectx)
654 return "%s@%s" % (self.path(), self._changectx)
651
655
652 def __repr__(self):
656 def __repr__(self):
653 return "<%s %s>" % (type(self).__name__, str(self))
657 return "<%s %s>" % (type(self).__name__, str(self))
654
658
655 def __hash__(self):
659 def __hash__(self):
656 try:
660 try:
657 return hash((self._path, self._filenode))
661 return hash((self._path, self._filenode))
658 except AttributeError:
662 except AttributeError:
659 return id(self)
663 return id(self)
660
664
661 def __eq__(self, other):
665 def __eq__(self, other):
662 try:
666 try:
663 return (type(self) == type(other) and self._path == other._path
667 return (type(self) == type(other) and self._path == other._path
664 and self._filenode == other._filenode)
668 and self._filenode == other._filenode)
665 except AttributeError:
669 except AttributeError:
666 return False
670 return False
667
671
668 def __ne__(self, other):
672 def __ne__(self, other):
669 return not (self == other)
673 return not (self == other)
670
674
671 def filerev(self):
675 def filerev(self):
672 return self._filerev
676 return self._filerev
673 def filenode(self):
677 def filenode(self):
674 return self._filenode
678 return self._filenode
675 def flags(self):
679 def flags(self):
676 return self._changectx.flags(self._path)
680 return self._changectx.flags(self._path)
677 def filelog(self):
681 def filelog(self):
678 return self._filelog
682 return self._filelog
679 def rev(self):
683 def rev(self):
680 return self._changeid
684 return self._changeid
681 def linkrev(self):
685 def linkrev(self):
682 return self._filelog.linkrev(self._filerev)
686 return self._filelog.linkrev(self._filerev)
683 def node(self):
687 def node(self):
684 return self._changectx.node()
688 return self._changectx.node()
685 def hex(self):
689 def hex(self):
686 return self._changectx.hex()
690 return self._changectx.hex()
687 def user(self):
691 def user(self):
688 return self._changectx.user()
692 return self._changectx.user()
689 def date(self):
693 def date(self):
690 return self._changectx.date()
694 return self._changectx.date()
691 def files(self):
695 def files(self):
692 return self._changectx.files()
696 return self._changectx.files()
693 def description(self):
697 def description(self):
694 return self._changectx.description()
698 return self._changectx.description()
695 def branch(self):
699 def branch(self):
696 return self._changectx.branch()
700 return self._changectx.branch()
697 def extra(self):
701 def extra(self):
698 return self._changectx.extra()
702 return self._changectx.extra()
699 def phase(self):
703 def phase(self):
700 return self._changectx.phase()
704 return self._changectx.phase()
701 def phasestr(self):
705 def phasestr(self):
702 return self._changectx.phasestr()
706 return self._changectx.phasestr()
703 def manifest(self):
707 def manifest(self):
704 return self._changectx.manifest()
708 return self._changectx.manifest()
705 def changectx(self):
709 def changectx(self):
706 return self._changectx
710 return self._changectx
707
711
708 def path(self):
712 def path(self):
709 return self._path
713 return self._path
710
714
711 def isbinary(self):
715 def isbinary(self):
712 try:
716 try:
713 return util.binary(self.data())
717 return util.binary(self.data())
714 except IOError:
718 except IOError:
715 return False
719 return False
716 def isexec(self):
720 def isexec(self):
717 return 'x' in self.flags()
721 return 'x' in self.flags()
718 def islink(self):
722 def islink(self):
719 return 'l' in self.flags()
723 return 'l' in self.flags()
720
724
721 def cmp(self, fctx):
725 def cmp(self, fctx):
722 """compare with other file context
726 """compare with other file context
723
727
724 returns True if different than fctx.
728 returns True if different than fctx.
725 """
729 """
726 if (fctx._filerev is None
730 if (fctx._filerev is None
727 and (self._repo._encodefilterpats
731 and (self._repo._encodefilterpats
728 # if file data starts with '\1\n', empty metadata block is
732 # if file data starts with '\1\n', empty metadata block is
729 # prepended, which adds 4 bytes to filelog.size().
733 # prepended, which adds 4 bytes to filelog.size().
730 or self.size() - 4 == fctx.size())
734 or self.size() - 4 == fctx.size())
731 or self.size() == fctx.size()):
735 or self.size() == fctx.size()):
732 return self._filelog.cmp(self._filenode, fctx.data())
736 return self._filelog.cmp(self._filenode, fctx.data())
733
737
734 return True
738 return True
735
739
736 def parents(self):
740 def parents(self):
737 _path = self._path
741 _path = self._path
738 fl = self._filelog
742 fl = self._filelog
739 pl = [(_path, n, fl) for n in self._filelog.parents(self._filenode)]
743 pl = [(_path, n, fl) for n in self._filelog.parents(self._filenode)]
740
744
741 r = self._filelog.renamed(self._filenode)
745 r = self._filelog.renamed(self._filenode)
742 if r:
746 if r:
743 pl[0] = (r[0], r[1], None)
747 pl[0] = (r[0], r[1], None)
744
748
745 return [filectx(self._repo, p, fileid=n, filelog=l)
749 return [filectx(self._repo, p, fileid=n, filelog=l)
746 for p, n, l in pl if n != nullid]
750 for p, n, l in pl if n != nullid]
747
751
748 def p1(self):
752 def p1(self):
749 return self.parents()[0]
753 return self.parents()[0]
750
754
751 def p2(self):
755 def p2(self):
752 p = self.parents()
756 p = self.parents()
753 if len(p) == 2:
757 if len(p) == 2:
754 return p[1]
758 return p[1]
755 return filectx(self._repo, self._path, fileid=-1, filelog=self._filelog)
759 return filectx(self._repo, self._path, fileid=-1, filelog=self._filelog)
756
760
757 def annotate(self, follow=False, linenumber=None, diffopts=None):
761 def annotate(self, follow=False, linenumber=None, diffopts=None):
758 '''returns a list of tuples of (ctx, line) for each line
762 '''returns a list of tuples of (ctx, line) for each line
759 in the file, where ctx is the filectx of the node where
763 in the file, where ctx is the filectx of the node where
760 that line was last changed.
764 that line was last changed.
761 This returns tuples of ((ctx, linenumber), line) for each line,
765 This returns tuples of ((ctx, linenumber), line) for each line,
762 if "linenumber" parameter is NOT "None".
766 if "linenumber" parameter is NOT "None".
763 In such tuples, linenumber means one at the first appearance
767 In such tuples, linenumber means one at the first appearance
764 in the managed file.
768 in the managed file.
765 To reduce annotation cost,
769 To reduce annotation cost,
766 this returns fixed value(False is used) as linenumber,
770 this returns fixed value(False is used) as linenumber,
767 if "linenumber" parameter is "False".'''
771 if "linenumber" parameter is "False".'''
768
772
769 if linenumber is None:
773 if linenumber is None:
770 def decorate(text, rev):
774 def decorate(text, rev):
771 return ([rev] * len(text.splitlines()), text)
775 return ([rev] * len(text.splitlines()), text)
772 elif linenumber:
776 elif linenumber:
773 def decorate(text, rev):
777 def decorate(text, rev):
774 size = len(text.splitlines())
778 size = len(text.splitlines())
775 return ([(rev, i) for i in xrange(1, size + 1)], text)
779 return ([(rev, i) for i in xrange(1, size + 1)], text)
776 else:
780 else:
777 def decorate(text, rev):
781 def decorate(text, rev):
778 return ([(rev, False)] * len(text.splitlines()), text)
782 return ([(rev, False)] * len(text.splitlines()), text)
779
783
780 def pair(parent, child):
784 def pair(parent, child):
781 blocks = mdiff.allblocks(parent[1], child[1], opts=diffopts,
785 blocks = mdiff.allblocks(parent[1], child[1], opts=diffopts,
782 refine=True)
786 refine=True)
783 for (a1, a2, b1, b2), t in blocks:
787 for (a1, a2, b1, b2), t in blocks:
784 # Changed blocks ('!') or blocks made only of blank lines ('~')
788 # Changed blocks ('!') or blocks made only of blank lines ('~')
785 # belong to the child.
789 # belong to the child.
786 if t == '=':
790 if t == '=':
787 child[0][b1:b2] = parent[0][a1:a2]
791 child[0][b1:b2] = parent[0][a1:a2]
788 return child
792 return child
789
793
790 getlog = util.lrucachefunc(lambda x: self._repo.file(x))
794 getlog = util.lrucachefunc(lambda x: self._repo.file(x))
791
795
792 def parents(f):
796 def parents(f):
793 pl = f.parents()
797 pl = f.parents()
794
798
795 # Don't return renamed parents if we aren't following.
799 # Don't return renamed parents if we aren't following.
796 if not follow:
800 if not follow:
797 pl = [p for p in pl if p.path() == f.path()]
801 pl = [p for p in pl if p.path() == f.path()]
798
802
799 # renamed filectx won't have a filelog yet, so set it
803 # renamed filectx won't have a filelog yet, so set it
800 # from the cache to save time
804 # from the cache to save time
801 for p in pl:
805 for p in pl:
802 if not '_filelog' in p.__dict__:
806 if not '_filelog' in p.__dict__:
803 p._filelog = getlog(p.path())
807 p._filelog = getlog(p.path())
804
808
805 return pl
809 return pl
806
810
807 # use linkrev to find the first changeset where self appeared
811 # use linkrev to find the first changeset where self appeared
808 if self.rev() != self.linkrev():
812 if self.rev() != self.linkrev():
809 base = self.filectx(self.filenode())
813 base = self.filectx(self.filenode())
810 else:
814 else:
811 base = self
815 base = self
812
816
813 # This algorithm would prefer to be recursive, but Python is a
817 # This algorithm would prefer to be recursive, but Python is a
814 # bit recursion-hostile. Instead we do an iterative
818 # bit recursion-hostile. Instead we do an iterative
815 # depth-first search.
819 # depth-first search.
816
820
817 visit = [base]
821 visit = [base]
818 hist = {}
822 hist = {}
819 pcache = {}
823 pcache = {}
820 needed = {base: 1}
824 needed = {base: 1}
821 while visit:
825 while visit:
822 f = visit[-1]
826 f = visit[-1]
823 pcached = f in pcache
827 pcached = f in pcache
824 if not pcached:
828 if not pcached:
825 pcache[f] = parents(f)
829 pcache[f] = parents(f)
826
830
827 ready = True
831 ready = True
828 pl = pcache[f]
832 pl = pcache[f]
829 for p in pl:
833 for p in pl:
830 if p not in hist:
834 if p not in hist:
831 ready = False
835 ready = False
832 visit.append(p)
836 visit.append(p)
833 if not pcached:
837 if not pcached:
834 needed[p] = needed.get(p, 0) + 1
838 needed[p] = needed.get(p, 0) + 1
835 if ready:
839 if ready:
836 visit.pop()
840 visit.pop()
837 reusable = f in hist
841 reusable = f in hist
838 if reusable:
842 if reusable:
839 curr = hist[f]
843 curr = hist[f]
840 else:
844 else:
841 curr = decorate(f.data(), f)
845 curr = decorate(f.data(), f)
842 for p in pl:
846 for p in pl:
843 if not reusable:
847 if not reusable:
844 curr = pair(hist[p], curr)
848 curr = pair(hist[p], curr)
845 if needed[p] == 1:
849 if needed[p] == 1:
846 del hist[p]
850 del hist[p]
847 del needed[p]
851 del needed[p]
848 else:
852 else:
849 needed[p] -= 1
853 needed[p] -= 1
850
854
851 hist[f] = curr
855 hist[f] = curr
852 pcache[f] = []
856 pcache[f] = []
853
857
854 return zip(hist[base][0], hist[base][1].splitlines(True))
858 return zip(hist[base][0], hist[base][1].splitlines(True))
855
859
856 def ancestors(self, followfirst=False):
860 def ancestors(self, followfirst=False):
857 visit = {}
861 visit = {}
858 c = self
862 c = self
859 cut = followfirst and 1 or None
863 cut = followfirst and 1 or None
860 while True:
864 while True:
861 for parent in c.parents()[:cut]:
865 for parent in c.parents()[:cut]:
862 visit[(parent.rev(), parent.node())] = parent
866 visit[(parent.rev(), parent.node())] = parent
863 if not visit:
867 if not visit:
864 break
868 break
865 c = visit.pop(max(visit))
869 c = visit.pop(max(visit))
866 yield c
870 yield c
867
871
868 class filectx(basefilectx):
872 class filectx(basefilectx):
869 """A filecontext object makes access to data related to a particular
873 """A filecontext object makes access to data related to a particular
870 filerevision convenient."""
874 filerevision convenient."""
871 def __init__(self, repo, path, changeid=None, fileid=None,
875 def __init__(self, repo, path, changeid=None, fileid=None,
872 filelog=None, changectx=None):
876 filelog=None, changectx=None):
873 """changeid can be a changeset revision, node, or tag.
877 """changeid can be a changeset revision, node, or tag.
874 fileid can be a file revision or node."""
878 fileid can be a file revision or node."""
875 self._repo = repo
879 self._repo = repo
876 self._path = path
880 self._path = path
877
881
878 assert (changeid is not None
882 assert (changeid is not None
879 or fileid is not None
883 or fileid is not None
880 or changectx is not None), \
884 or changectx is not None), \
881 ("bad args: changeid=%r, fileid=%r, changectx=%r"
885 ("bad args: changeid=%r, fileid=%r, changectx=%r"
882 % (changeid, fileid, changectx))
886 % (changeid, fileid, changectx))
883
887
884 if filelog is not None:
888 if filelog is not None:
885 self._filelog = filelog
889 self._filelog = filelog
886
890
887 if changeid is not None:
891 if changeid is not None:
888 self._changeid = changeid
892 self._changeid = changeid
889 if changectx is not None:
893 if changectx is not None:
890 self._changectx = changectx
894 self._changectx = changectx
891 if fileid is not None:
895 if fileid is not None:
892 self._fileid = fileid
896 self._fileid = fileid
893
897
894 @propertycache
898 @propertycache
895 def _changectx(self):
899 def _changectx(self):
896 try:
900 try:
897 return changectx(self._repo, self._changeid)
901 return changectx(self._repo, self._changeid)
898 except error.RepoLookupError:
902 except error.RepoLookupError:
899 # Linkrev may point to any revision in the repository. When the
903 # Linkrev may point to any revision in the repository. When the
900 # repository is filtered this may lead to `filectx` trying to build
904 # repository is filtered this may lead to `filectx` trying to build
901 # `changectx` for filtered revision. In such case we fallback to
905 # `changectx` for filtered revision. In such case we fallback to
902 # creating `changectx` on the unfiltered version of the reposition.
906 # creating `changectx` on the unfiltered version of the reposition.
903 # This fallback should not be an issue because `changectx` from
907 # This fallback should not be an issue because `changectx` from
904 # `filectx` are not used in complex operations that care about
908 # `filectx` are not used in complex operations that care about
905 # filtering.
909 # filtering.
906 #
910 #
907 # This fallback is a cheap and dirty fix that prevent several
911 # This fallback is a cheap and dirty fix that prevent several
908 # crashes. It does not ensure the behavior is correct. However the
912 # crashes. It does not ensure the behavior is correct. However the
909 # behavior was not correct before filtering either and "incorrect
913 # behavior was not correct before filtering either and "incorrect
910 # behavior" is seen as better as "crash"
914 # behavior" is seen as better as "crash"
911 #
915 #
912 # Linkrevs have several serious troubles with filtering that are
916 # Linkrevs have several serious troubles with filtering that are
913 # complicated to solve. Proper handling of the issue here should be
917 # complicated to solve. Proper handling of the issue here should be
914 # considered when solving linkrev issue are on the table.
918 # considered when solving linkrev issue are on the table.
915 return changectx(self._repo.unfiltered(), self._changeid)
919 return changectx(self._repo.unfiltered(), self._changeid)
916
920
917 def filectx(self, fileid):
921 def filectx(self, fileid):
918 '''opens an arbitrary revision of the file without
922 '''opens an arbitrary revision of the file without
919 opening a new filelog'''
923 opening a new filelog'''
920 return filectx(self._repo, self._path, fileid=fileid,
924 return filectx(self._repo, self._path, fileid=fileid,
921 filelog=self._filelog)
925 filelog=self._filelog)
922
926
923 def data(self):
927 def data(self):
924 try:
928 try:
925 return self._filelog.read(self._filenode)
929 return self._filelog.read(self._filenode)
926 except error.CensoredNodeError:
930 except error.CensoredNodeError:
927 if self._repo.ui.config("censor", "policy", "abort") == "ignore":
931 if self._repo.ui.config("censor", "policy", "abort") == "ignore":
928 return ""
932 return ""
929 raise util.Abort(_("censored node: %s") % short(self._filenode),
933 raise util.Abort(_("censored node: %s") % short(self._filenode),
930 hint=_("set censor.policy to ignore errors"))
934 hint=_("set censor.policy to ignore errors"))
931
935
932 def size(self):
936 def size(self):
933 return self._filelog.size(self._filerev)
937 return self._filelog.size(self._filerev)
934
938
935 def renamed(self):
939 def renamed(self):
936 """check if file was actually renamed in this changeset revision
940 """check if file was actually renamed in this changeset revision
937
941
938 If rename logged in file revision, we report copy for changeset only
942 If rename logged in file revision, we report copy for changeset only
939 if file revisions linkrev points back to the changeset in question
943 if file revisions linkrev points back to the changeset in question
940 or both changeset parents contain different file revisions.
944 or both changeset parents contain different file revisions.
941 """
945 """
942
946
943 renamed = self._filelog.renamed(self._filenode)
947 renamed = self._filelog.renamed(self._filenode)
944 if not renamed:
948 if not renamed:
945 return renamed
949 return renamed
946
950
947 if self.rev() == self.linkrev():
951 if self.rev() == self.linkrev():
948 return renamed
952 return renamed
949
953
950 name = self.path()
954 name = self.path()
951 fnode = self._filenode
955 fnode = self._filenode
952 for p in self._changectx.parents():
956 for p in self._changectx.parents():
953 try:
957 try:
954 if fnode == p.filenode(name):
958 if fnode == p.filenode(name):
955 return None
959 return None
956 except error.LookupError:
960 except error.LookupError:
957 pass
961 pass
958 return renamed
962 return renamed
959
963
960 def children(self):
964 def children(self):
961 # hard for renames
965 # hard for renames
962 c = self._filelog.children(self._filenode)
966 c = self._filelog.children(self._filenode)
963 return [filectx(self._repo, self._path, fileid=x,
967 return [filectx(self._repo, self._path, fileid=x,
964 filelog=self._filelog) for x in c]
968 filelog=self._filelog) for x in c]
965
969
966 class committablectx(basectx):
970 class committablectx(basectx):
967 """A committablectx object provides common functionality for a context that
971 """A committablectx object provides common functionality for a context that
968 wants the ability to commit, e.g. workingctx or memctx."""
972 wants the ability to commit, e.g. workingctx or memctx."""
969 def __init__(self, repo, text="", user=None, date=None, extra=None,
973 def __init__(self, repo, text="", user=None, date=None, extra=None,
970 changes=None):
974 changes=None):
971 self._repo = repo
975 self._repo = repo
972 self._rev = None
976 self._rev = None
973 self._node = None
977 self._node = None
974 self._text = text
978 self._text = text
975 if date:
979 if date:
976 self._date = util.parsedate(date)
980 self._date = util.parsedate(date)
977 if user:
981 if user:
978 self._user = user
982 self._user = user
979 if changes:
983 if changes:
980 self._status = changes
984 self._status = changes
981
985
982 self._extra = {}
986 self._extra = {}
983 if extra:
987 if extra:
984 self._extra = extra.copy()
988 self._extra = extra.copy()
985 if 'branch' not in self._extra:
989 if 'branch' not in self._extra:
986 try:
990 try:
987 branch = encoding.fromlocal(self._repo.dirstate.branch())
991 branch = encoding.fromlocal(self._repo.dirstate.branch())
988 except UnicodeDecodeError:
992 except UnicodeDecodeError:
989 raise util.Abort(_('branch name not in UTF-8!'))
993 raise util.Abort(_('branch name not in UTF-8!'))
990 self._extra['branch'] = branch
994 self._extra['branch'] = branch
991 if self._extra['branch'] == '':
995 if self._extra['branch'] == '':
992 self._extra['branch'] = 'default'
996 self._extra['branch'] = 'default'
993
997
994 def __str__(self):
998 def __str__(self):
995 return str(self._parents[0]) + "+"
999 return str(self._parents[0]) + "+"
996
1000
997 def __nonzero__(self):
1001 def __nonzero__(self):
998 return True
1002 return True
999
1003
1000 def _buildflagfunc(self):
1004 def _buildflagfunc(self):
1001 # Create a fallback function for getting file flags when the
1005 # Create a fallback function for getting file flags when the
1002 # filesystem doesn't support them
1006 # filesystem doesn't support them
1003
1007
1004 copiesget = self._repo.dirstate.copies().get
1008 copiesget = self._repo.dirstate.copies().get
1005
1009
1006 if len(self._parents) < 2:
1010 if len(self._parents) < 2:
1007 # when we have one parent, it's easy: copy from parent
1011 # when we have one parent, it's easy: copy from parent
1008 man = self._parents[0].manifest()
1012 man = self._parents[0].manifest()
1009 def func(f):
1013 def func(f):
1010 f = copiesget(f, f)
1014 f = copiesget(f, f)
1011 return man.flags(f)
1015 return man.flags(f)
1012 else:
1016 else:
1013 # merges are tricky: we try to reconstruct the unstored
1017 # merges are tricky: we try to reconstruct the unstored
1014 # result from the merge (issue1802)
1018 # result from the merge (issue1802)
1015 p1, p2 = self._parents
1019 p1, p2 = self._parents
1016 pa = p1.ancestor(p2)
1020 pa = p1.ancestor(p2)
1017 m1, m2, ma = p1.manifest(), p2.manifest(), pa.manifest()
1021 m1, m2, ma = p1.manifest(), p2.manifest(), pa.manifest()
1018
1022
1019 def func(f):
1023 def func(f):
1020 f = copiesget(f, f) # may be wrong for merges with copies
1024 f = copiesget(f, f) # may be wrong for merges with copies
1021 fl1, fl2, fla = m1.flags(f), m2.flags(f), ma.flags(f)
1025 fl1, fl2, fla = m1.flags(f), m2.flags(f), ma.flags(f)
1022 if fl1 == fl2:
1026 if fl1 == fl2:
1023 return fl1
1027 return fl1
1024 if fl1 == fla:
1028 if fl1 == fla:
1025 return fl2
1029 return fl2
1026 if fl2 == fla:
1030 if fl2 == fla:
1027 return fl1
1031 return fl1
1028 return '' # punt for conflicts
1032 return '' # punt for conflicts
1029
1033
1030 return func
1034 return func
1031
1035
1032 @propertycache
1036 @propertycache
1033 def _flagfunc(self):
1037 def _flagfunc(self):
1034 return self._repo.dirstate.flagfunc(self._buildflagfunc)
1038 return self._repo.dirstate.flagfunc(self._buildflagfunc)
1035
1039
1036 @propertycache
1040 @propertycache
1037 def _manifest(self):
1041 def _manifest(self):
1038 """generate a manifest corresponding to the values in self._status
1042 """generate a manifest corresponding to the values in self._status
1039
1043
1040 This reuse the file nodeid from parent, but we append an extra letter
1044 This reuse the file nodeid from parent, but we append an extra letter
1041 when modified. Modified files get an extra 'm' while added files get
1045 when modified. Modified files get an extra 'm' while added files get
1042 an extra 'a'. This is used by manifests merge to see that files
1046 an extra 'a'. This is used by manifests merge to see that files
1043 are different and by update logic to avoid deleting newly added files.
1047 are different and by update logic to avoid deleting newly added files.
1044 """
1048 """
1045
1049
1046 man1 = self._parents[0].manifest()
1050 man1 = self._parents[0].manifest()
1047 man = man1.copy()
1051 man = man1.copy()
1048 if len(self._parents) > 1:
1052 if len(self._parents) > 1:
1049 man2 = self.p2().manifest()
1053 man2 = self.p2().manifest()
1050 def getman(f):
1054 def getman(f):
1051 if f in man1:
1055 if f in man1:
1052 return man1
1056 return man1
1053 return man2
1057 return man2
1054 else:
1058 else:
1055 getman = lambda f: man1
1059 getman = lambda f: man1
1056
1060
1057 copied = self._repo.dirstate.copies()
1061 copied = self._repo.dirstate.copies()
1058 ff = self._flagfunc
1062 ff = self._flagfunc
1059 for i, l in (("a", self._status.added), ("m", self._status.modified)):
1063 for i, l in (("a", self._status.added), ("m", self._status.modified)):
1060 for f in l:
1064 for f in l:
1061 orig = copied.get(f, f)
1065 orig = copied.get(f, f)
1062 man[f] = getman(orig).get(orig, nullid) + i
1066 man[f] = getman(orig).get(orig, nullid) + i
1063 try:
1067 try:
1064 man.setflag(f, ff(f))
1068 man.setflag(f, ff(f))
1065 except OSError:
1069 except OSError:
1066 pass
1070 pass
1067
1071
1068 for f in self._status.deleted + self._status.removed:
1072 for f in self._status.deleted + self._status.removed:
1069 if f in man:
1073 if f in man:
1070 del man[f]
1074 del man[f]
1071
1075
1072 return man
1076 return man
1073
1077
1074 @propertycache
1078 @propertycache
1075 def _status(self):
1079 def _status(self):
1076 return self._repo.status()
1080 return self._repo.status()
1077
1081
1078 @propertycache
1082 @propertycache
1079 def _user(self):
1083 def _user(self):
1080 return self._repo.ui.username()
1084 return self._repo.ui.username()
1081
1085
1082 @propertycache
1086 @propertycache
1083 def _date(self):
1087 def _date(self):
1084 return util.makedate()
1088 return util.makedate()
1085
1089
1086 def subrev(self, subpath):
1090 def subrev(self, subpath):
1087 return None
1091 return None
1088
1092
1089 def user(self):
1093 def user(self):
1090 return self._user or self._repo.ui.username()
1094 return self._user or self._repo.ui.username()
1091 def date(self):
1095 def date(self):
1092 return self._date
1096 return self._date
1093 def description(self):
1097 def description(self):
1094 return self._text
1098 return self._text
1095 def files(self):
1099 def files(self):
1096 return sorted(self._status.modified + self._status.added +
1100 return sorted(self._status.modified + self._status.added +
1097 self._status.removed)
1101 self._status.removed)
1098
1102
1099 def modified(self):
1103 def modified(self):
1100 return self._status.modified
1104 return self._status.modified
1101 def added(self):
1105 def added(self):
1102 return self._status.added
1106 return self._status.added
1103 def removed(self):
1107 def removed(self):
1104 return self._status.removed
1108 return self._status.removed
1105 def deleted(self):
1109 def deleted(self):
1106 return self._status.deleted
1110 return self._status.deleted
1107 def unknown(self):
1111 def unknown(self):
1108 return self._status.unknown
1112 return self._status.unknown
1109 def ignored(self):
1113 def ignored(self):
1110 return self._status.ignored
1114 return self._status.ignored
1111 def clean(self):
1115 def clean(self):
1112 return self._status.clean
1116 return self._status.clean
1113 def branch(self):
1117 def branch(self):
1114 return encoding.tolocal(self._extra['branch'])
1118 return encoding.tolocal(self._extra['branch'])
1115 def closesbranch(self):
1119 def closesbranch(self):
1116 return 'close' in self._extra
1120 return 'close' in self._extra
1117 def extra(self):
1121 def extra(self):
1118 return self._extra
1122 return self._extra
1119
1123
1120 def tags(self):
1124 def tags(self):
1121 t = []
1125 t = []
1122 for p in self.parents():
1126 for p in self.parents():
1123 t.extend(p.tags())
1127 t.extend(p.tags())
1124 return t
1128 return t
1125
1129
1126 def bookmarks(self):
1130 def bookmarks(self):
1127 b = []
1131 b = []
1128 for p in self.parents():
1132 for p in self.parents():
1129 b.extend(p.bookmarks())
1133 b.extend(p.bookmarks())
1130 return b
1134 return b
1131
1135
1132 def phase(self):
1136 def phase(self):
1133 phase = phases.draft # default phase to draft
1137 phase = phases.draft # default phase to draft
1134 for p in self.parents():
1138 for p in self.parents():
1135 phase = max(phase, p.phase())
1139 phase = max(phase, p.phase())
1136 return phase
1140 return phase
1137
1141
1138 def hidden(self):
1142 def hidden(self):
1139 return False
1143 return False
1140
1144
1141 def children(self):
1145 def children(self):
1142 return []
1146 return []
1143
1147
1144 def flags(self, path):
1148 def flags(self, path):
1145 if '_manifest' in self.__dict__:
1149 if '_manifest' in self.__dict__:
1146 try:
1150 try:
1147 return self._manifest.flags(path)
1151 return self._manifest.flags(path)
1148 except KeyError:
1152 except KeyError:
1149 return ''
1153 return ''
1150
1154
1151 try:
1155 try:
1152 return self._flagfunc(path)
1156 return self._flagfunc(path)
1153 except OSError:
1157 except OSError:
1154 return ''
1158 return ''
1155
1159
1156 def ancestor(self, c2):
1160 def ancestor(self, c2):
1157 """return the "best" ancestor context of self and c2"""
1161 """return the "best" ancestor context of self and c2"""
1158 return self._parents[0].ancestor(c2) # punt on two parents for now
1162 return self._parents[0].ancestor(c2) # punt on two parents for now
1159
1163
1160 def walk(self, match):
1164 def walk(self, match):
1161 return sorted(self._repo.dirstate.walk(match, sorted(self.substate),
1165 return sorted(self._repo.dirstate.walk(match, sorted(self.substate),
1162 True, False))
1166 True, False))
1163
1167
1164 def matches(self, match):
1168 def matches(self, match):
1165 return sorted(self._repo.dirstate.matches(match))
1169 return sorted(self._repo.dirstate.matches(match))
1166
1170
1167 def ancestors(self):
1171 def ancestors(self):
1168 for a in self._repo.changelog.ancestors(
1172 for a in self._repo.changelog.ancestors(
1169 [p.rev() for p in self._parents]):
1173 [p.rev() for p in self._parents]):
1170 yield changectx(self._repo, a)
1174 yield changectx(self._repo, a)
1171
1175
1172 def markcommitted(self, node):
1176 def markcommitted(self, node):
1173 """Perform post-commit cleanup necessary after committing this ctx
1177 """Perform post-commit cleanup necessary after committing this ctx
1174
1178
1175 Specifically, this updates backing stores this working context
1179 Specifically, this updates backing stores this working context
1176 wraps to reflect the fact that the changes reflected by this
1180 wraps to reflect the fact that the changes reflected by this
1177 workingctx have been committed. For example, it marks
1181 workingctx have been committed. For example, it marks
1178 modified and added files as normal in the dirstate.
1182 modified and added files as normal in the dirstate.
1179
1183
1180 """
1184 """
1181
1185
1182 self._repo.dirstate.beginparentchange()
1186 self._repo.dirstate.beginparentchange()
1183 for f in self.modified() + self.added():
1187 for f in self.modified() + self.added():
1184 self._repo.dirstate.normal(f)
1188 self._repo.dirstate.normal(f)
1185 for f in self.removed():
1189 for f in self.removed():
1186 self._repo.dirstate.drop(f)
1190 self._repo.dirstate.drop(f)
1187 self._repo.dirstate.setparents(node)
1191 self._repo.dirstate.setparents(node)
1188 self._repo.dirstate.endparentchange()
1192 self._repo.dirstate.endparentchange()
1189
1193
1190 def dirs(self):
1194 def dirs(self):
1191 return self._repo.dirstate.dirs()
1195 return self._repo.dirstate.dirs()
1192
1196
1193 class workingctx(committablectx):
1197 class workingctx(committablectx):
1194 """A workingctx object makes access to data related to
1198 """A workingctx object makes access to data related to
1195 the current working directory convenient.
1199 the current working directory convenient.
1196 date - any valid date string or (unixtime, offset), or None.
1200 date - any valid date string or (unixtime, offset), or None.
1197 user - username string, or None.
1201 user - username string, or None.
1198 extra - a dictionary of extra values, or None.
1202 extra - a dictionary of extra values, or None.
1199 changes - a list of file lists as returned by localrepo.status()
1203 changes - a list of file lists as returned by localrepo.status()
1200 or None to use the repository status.
1204 or None to use the repository status.
1201 """
1205 """
1202 def __init__(self, repo, text="", user=None, date=None, extra=None,
1206 def __init__(self, repo, text="", user=None, date=None, extra=None,
1203 changes=None):
1207 changes=None):
1204 super(workingctx, self).__init__(repo, text, user, date, extra, changes)
1208 super(workingctx, self).__init__(repo, text, user, date, extra, changes)
1205
1209
1206 def __iter__(self):
1210 def __iter__(self):
1207 d = self._repo.dirstate
1211 d = self._repo.dirstate
1208 for f in d:
1212 for f in d:
1209 if d[f] != 'r':
1213 if d[f] != 'r':
1210 yield f
1214 yield f
1211
1215
1212 def __contains__(self, key):
1216 def __contains__(self, key):
1213 return self._repo.dirstate[key] not in "?r"
1217 return self._repo.dirstate[key] not in "?r"
1214
1218
1215 @propertycache
1219 @propertycache
1216 def _parents(self):
1220 def _parents(self):
1217 p = self._repo.dirstate.parents()
1221 p = self._repo.dirstate.parents()
1218 if p[1] == nullid:
1222 if p[1] == nullid:
1219 p = p[:-1]
1223 p = p[:-1]
1220 return [changectx(self._repo, x) for x in p]
1224 return [changectx(self._repo, x) for x in p]
1221
1225
1222 def filectx(self, path, filelog=None):
1226 def filectx(self, path, filelog=None):
1223 """get a file context from the working directory"""
1227 """get a file context from the working directory"""
1224 return workingfilectx(self._repo, path, workingctx=self,
1228 return workingfilectx(self._repo, path, workingctx=self,
1225 filelog=filelog)
1229 filelog=filelog)
1226
1230
1227 def dirty(self, missing=False, merge=True, branch=True):
1231 def dirty(self, missing=False, merge=True, branch=True):
1228 "check whether a working directory is modified"
1232 "check whether a working directory is modified"
1229 # check subrepos first
1233 # check subrepos first
1230 for s in sorted(self.substate):
1234 for s in sorted(self.substate):
1231 if self.sub(s).dirty():
1235 if self.sub(s).dirty():
1232 return True
1236 return True
1233 # check current working dir
1237 # check current working dir
1234 return ((merge and self.p2()) or
1238 return ((merge and self.p2()) or
1235 (branch and self.branch() != self.p1().branch()) or
1239 (branch and self.branch() != self.p1().branch()) or
1236 self.modified() or self.added() or self.removed() or
1240 self.modified() or self.added() or self.removed() or
1237 (missing and self.deleted()))
1241 (missing and self.deleted()))
1238
1242
1239 def add(self, list, prefix=""):
1243 def add(self, list, prefix=""):
1240 join = lambda f: os.path.join(prefix, f)
1244 join = lambda f: os.path.join(prefix, f)
1241 wlock = self._repo.wlock()
1245 wlock = self._repo.wlock()
1242 ui, ds = self._repo.ui, self._repo.dirstate
1246 ui, ds = self._repo.ui, self._repo.dirstate
1243 try:
1247 try:
1244 rejected = []
1248 rejected = []
1245 lstat = self._repo.wvfs.lstat
1249 lstat = self._repo.wvfs.lstat
1246 for f in list:
1250 for f in list:
1247 scmutil.checkportable(ui, join(f))
1251 scmutil.checkportable(ui, join(f))
1248 try:
1252 try:
1249 st = lstat(f)
1253 st = lstat(f)
1250 except OSError:
1254 except OSError:
1251 ui.warn(_("%s does not exist!\n") % join(f))
1255 ui.warn(_("%s does not exist!\n") % join(f))
1252 rejected.append(f)
1256 rejected.append(f)
1253 continue
1257 continue
1254 if st.st_size > 10000000:
1258 if st.st_size > 10000000:
1255 ui.warn(_("%s: up to %d MB of RAM may be required "
1259 ui.warn(_("%s: up to %d MB of RAM may be required "
1256 "to manage this file\n"
1260 "to manage this file\n"
1257 "(use 'hg revert %s' to cancel the "
1261 "(use 'hg revert %s' to cancel the "
1258 "pending addition)\n")
1262 "pending addition)\n")
1259 % (f, 3 * st.st_size // 1000000, join(f)))
1263 % (f, 3 * st.st_size // 1000000, join(f)))
1260 if not (stat.S_ISREG(st.st_mode) or stat.S_ISLNK(st.st_mode)):
1264 if not (stat.S_ISREG(st.st_mode) or stat.S_ISLNK(st.st_mode)):
1261 ui.warn(_("%s not added: only files and symlinks "
1265 ui.warn(_("%s not added: only files and symlinks "
1262 "supported currently\n") % join(f))
1266 "supported currently\n") % join(f))
1263 rejected.append(f)
1267 rejected.append(f)
1264 elif ds[f] in 'amn':
1268 elif ds[f] in 'amn':
1265 ui.warn(_("%s already tracked!\n") % join(f))
1269 ui.warn(_("%s already tracked!\n") % join(f))
1266 elif ds[f] == 'r':
1270 elif ds[f] == 'r':
1267 ds.normallookup(f)
1271 ds.normallookup(f)
1268 else:
1272 else:
1269 ds.add(f)
1273 ds.add(f)
1270 return rejected
1274 return rejected
1271 finally:
1275 finally:
1272 wlock.release()
1276 wlock.release()
1273
1277
1274 def forget(self, files, prefix=""):
1278 def forget(self, files, prefix=""):
1275 join = lambda f: os.path.join(prefix, f)
1279 join = lambda f: os.path.join(prefix, f)
1276 wlock = self._repo.wlock()
1280 wlock = self._repo.wlock()
1277 try:
1281 try:
1278 rejected = []
1282 rejected = []
1279 for f in files:
1283 for f in files:
1280 if f not in self._repo.dirstate:
1284 if f not in self._repo.dirstate:
1281 self._repo.ui.warn(_("%s not tracked!\n") % join(f))
1285 self._repo.ui.warn(_("%s not tracked!\n") % join(f))
1282 rejected.append(f)
1286 rejected.append(f)
1283 elif self._repo.dirstate[f] != 'a':
1287 elif self._repo.dirstate[f] != 'a':
1284 self._repo.dirstate.remove(f)
1288 self._repo.dirstate.remove(f)
1285 else:
1289 else:
1286 self._repo.dirstate.drop(f)
1290 self._repo.dirstate.drop(f)
1287 return rejected
1291 return rejected
1288 finally:
1292 finally:
1289 wlock.release()
1293 wlock.release()
1290
1294
1291 def undelete(self, list):
1295 def undelete(self, list):
1292 pctxs = self.parents()
1296 pctxs = self.parents()
1293 wlock = self._repo.wlock()
1297 wlock = self._repo.wlock()
1294 try:
1298 try:
1295 for f in list:
1299 for f in list:
1296 if self._repo.dirstate[f] != 'r':
1300 if self._repo.dirstate[f] != 'r':
1297 self._repo.ui.warn(_("%s not removed!\n") % f)
1301 self._repo.ui.warn(_("%s not removed!\n") % f)
1298 else:
1302 else:
1299 fctx = f in pctxs[0] and pctxs[0][f] or pctxs[1][f]
1303 fctx = f in pctxs[0] and pctxs[0][f] or pctxs[1][f]
1300 t = fctx.data()
1304 t = fctx.data()
1301 self._repo.wwrite(f, t, fctx.flags())
1305 self._repo.wwrite(f, t, fctx.flags())
1302 self._repo.dirstate.normal(f)
1306 self._repo.dirstate.normal(f)
1303 finally:
1307 finally:
1304 wlock.release()
1308 wlock.release()
1305
1309
1306 def copy(self, source, dest):
1310 def copy(self, source, dest):
1307 try:
1311 try:
1308 st = self._repo.wvfs.lstat(dest)
1312 st = self._repo.wvfs.lstat(dest)
1309 except OSError, err:
1313 except OSError, err:
1310 if err.errno != errno.ENOENT:
1314 if err.errno != errno.ENOENT:
1311 raise
1315 raise
1312 self._repo.ui.warn(_("%s does not exist!\n") % dest)
1316 self._repo.ui.warn(_("%s does not exist!\n") % dest)
1313 return
1317 return
1314 if not (stat.S_ISREG(st.st_mode) or stat.S_ISLNK(st.st_mode)):
1318 if not (stat.S_ISREG(st.st_mode) or stat.S_ISLNK(st.st_mode)):
1315 self._repo.ui.warn(_("copy failed: %s is not a file or a "
1319 self._repo.ui.warn(_("copy failed: %s is not a file or a "
1316 "symbolic link\n") % dest)
1320 "symbolic link\n") % dest)
1317 else:
1321 else:
1318 wlock = self._repo.wlock()
1322 wlock = self._repo.wlock()
1319 try:
1323 try:
1320 if self._repo.dirstate[dest] in '?':
1324 if self._repo.dirstate[dest] in '?':
1321 self._repo.dirstate.add(dest)
1325 self._repo.dirstate.add(dest)
1322 elif self._repo.dirstate[dest] in 'r':
1326 elif self._repo.dirstate[dest] in 'r':
1323 self._repo.dirstate.normallookup(dest)
1327 self._repo.dirstate.normallookup(dest)
1324 self._repo.dirstate.copy(source, dest)
1328 self._repo.dirstate.copy(source, dest)
1325 finally:
1329 finally:
1326 wlock.release()
1330 wlock.release()
1327
1331
1328 def _filtersuspectsymlink(self, files):
1332 def _filtersuspectsymlink(self, files):
1329 if not files or self._repo.dirstate._checklink:
1333 if not files or self._repo.dirstate._checklink:
1330 return files
1334 return files
1331
1335
1332 # Symlink placeholders may get non-symlink-like contents
1336 # Symlink placeholders may get non-symlink-like contents
1333 # via user error or dereferencing by NFS or Samba servers,
1337 # via user error or dereferencing by NFS or Samba servers,
1334 # so we filter out any placeholders that don't look like a
1338 # so we filter out any placeholders that don't look like a
1335 # symlink
1339 # symlink
1336 sane = []
1340 sane = []
1337 for f in files:
1341 for f in files:
1338 if self.flags(f) == 'l':
1342 if self.flags(f) == 'l':
1339 d = self[f].data()
1343 d = self[f].data()
1340 if d == '' or len(d) >= 1024 or '\n' in d or util.binary(d):
1344 if d == '' or len(d) >= 1024 or '\n' in d or util.binary(d):
1341 self._repo.ui.debug('ignoring suspect symlink placeholder'
1345 self._repo.ui.debug('ignoring suspect symlink placeholder'
1342 ' "%s"\n' % f)
1346 ' "%s"\n' % f)
1343 continue
1347 continue
1344 sane.append(f)
1348 sane.append(f)
1345 return sane
1349 return sane
1346
1350
1347 def _checklookup(self, files):
1351 def _checklookup(self, files):
1348 # check for any possibly clean files
1352 # check for any possibly clean files
1349 if not files:
1353 if not files:
1350 return [], []
1354 return [], []
1351
1355
1352 modified = []
1356 modified = []
1353 fixup = []
1357 fixup = []
1354 pctx = self._parents[0]
1358 pctx = self._parents[0]
1355 # do a full compare of any files that might have changed
1359 # do a full compare of any files that might have changed
1356 for f in sorted(files):
1360 for f in sorted(files):
1357 if (f not in pctx or self.flags(f) != pctx.flags(f)
1361 if (f not in pctx or self.flags(f) != pctx.flags(f)
1358 or pctx[f].cmp(self[f])):
1362 or pctx[f].cmp(self[f])):
1359 modified.append(f)
1363 modified.append(f)
1360 else:
1364 else:
1361 fixup.append(f)
1365 fixup.append(f)
1362
1366
1363 # update dirstate for files that are actually clean
1367 # update dirstate for files that are actually clean
1364 if fixup:
1368 if fixup:
1365 try:
1369 try:
1366 # updating the dirstate is optional
1370 # updating the dirstate is optional
1367 # so we don't wait on the lock
1371 # so we don't wait on the lock
1368 # wlock can invalidate the dirstate, so cache normal _after_
1372 # wlock can invalidate the dirstate, so cache normal _after_
1369 # taking the lock
1373 # taking the lock
1370 wlock = self._repo.wlock(False)
1374 wlock = self._repo.wlock(False)
1371 normal = self._repo.dirstate.normal
1375 normal = self._repo.dirstate.normal
1372 try:
1376 try:
1373 for f in fixup:
1377 for f in fixup:
1374 normal(f)
1378 normal(f)
1375 finally:
1379 finally:
1376 wlock.release()
1380 wlock.release()
1377 except error.LockError:
1381 except error.LockError:
1378 pass
1382 pass
1379 return modified, fixup
1383 return modified, fixup
1380
1384
1381 def _manifestmatches(self, match, s):
1385 def _manifestmatches(self, match, s):
1382 """Slow path for workingctx
1386 """Slow path for workingctx
1383
1387
1384 The fast path is when we compare the working directory to its parent
1388 The fast path is when we compare the working directory to its parent
1385 which means this function is comparing with a non-parent; therefore we
1389 which means this function is comparing with a non-parent; therefore we
1386 need to build a manifest and return what matches.
1390 need to build a manifest and return what matches.
1387 """
1391 """
1388 mf = self._repo['.']._manifestmatches(match, s)
1392 mf = self._repo['.']._manifestmatches(match, s)
1389 for f in s.modified + s.added:
1393 for f in s.modified + s.added:
1390 mf[f] = None
1394 mf[f] = None
1391 mf.setflag(f, self.flags(f))
1395 mf.setflag(f, self.flags(f))
1392 for f in s.removed:
1396 for f in s.removed:
1393 if f in mf:
1397 if f in mf:
1394 del mf[f]
1398 del mf[f]
1395 return mf
1399 return mf
1396
1400
1397 def _dirstatestatus(self, match=None, ignored=False, clean=False,
1401 def _dirstatestatus(self, match=None, ignored=False, clean=False,
1398 unknown=False):
1402 unknown=False):
1399 '''Gets the status from the dirstate -- internal use only.'''
1403 '''Gets the status from the dirstate -- internal use only.'''
1400 listignored, listclean, listunknown = ignored, clean, unknown
1404 listignored, listclean, listunknown = ignored, clean, unknown
1401 match = match or matchmod.always(self._repo.root, self._repo.getcwd())
1405 match = match or matchmod.always(self._repo.root, self._repo.getcwd())
1402 subrepos = []
1406 subrepos = []
1403 if '.hgsub' in self:
1407 if '.hgsub' in self:
1404 subrepos = sorted(self.substate)
1408 subrepos = sorted(self.substate)
1405 cmp, s = self._repo.dirstate.status(match, subrepos, listignored,
1409 cmp, s = self._repo.dirstate.status(match, subrepos, listignored,
1406 listclean, listunknown)
1410 listclean, listunknown)
1407
1411
1408 # check for any possibly clean files
1412 # check for any possibly clean files
1409 if cmp:
1413 if cmp:
1410 modified2, fixup = self._checklookup(cmp)
1414 modified2, fixup = self._checklookup(cmp)
1411 s.modified.extend(modified2)
1415 s.modified.extend(modified2)
1412
1416
1413 # update dirstate for files that are actually clean
1417 # update dirstate for files that are actually clean
1414 if fixup and listclean:
1418 if fixup and listclean:
1415 s.clean.extend(fixup)
1419 s.clean.extend(fixup)
1416
1420
1417 return s
1421 return s
1418
1422
1419 def _buildstatus(self, other, s, match, listignored, listclean,
1423 def _buildstatus(self, other, s, match, listignored, listclean,
1420 listunknown):
1424 listunknown):
1421 """build a status with respect to another context
1425 """build a status with respect to another context
1422
1426
1423 This includes logic for maintaining the fast path of status when
1427 This includes logic for maintaining the fast path of status when
1424 comparing the working directory against its parent, which is to skip
1428 comparing the working directory against its parent, which is to skip
1425 building a new manifest if self (working directory) is not comparing
1429 building a new manifest if self (working directory) is not comparing
1426 against its parent (repo['.']).
1430 against its parent (repo['.']).
1427 """
1431 """
1428 s = self._dirstatestatus(match, listignored, listclean, listunknown)
1432 s = self._dirstatestatus(match, listignored, listclean, listunknown)
1429 # Filter out symlinks that, in the case of FAT32 and NTFS filesystems,
1433 # Filter out symlinks that, in the case of FAT32 and NTFS filesystems,
1430 # might have accidentally ended up with the entire contents of the file
1434 # might have accidentally ended up with the entire contents of the file
1431 # they are supposed to be linking to.
1435 # they are supposed to be linking to.
1432 s.modified[:] = self._filtersuspectsymlink(s.modified)
1436 s.modified[:] = self._filtersuspectsymlink(s.modified)
1433 if other != self._repo['.']:
1437 if other != self._repo['.']:
1434 s = super(workingctx, self)._buildstatus(other, s, match,
1438 s = super(workingctx, self)._buildstatus(other, s, match,
1435 listignored, listclean,
1439 listignored, listclean,
1436 listunknown)
1440 listunknown)
1437 self._status = s
1441 self._status = s
1438 return s
1442 return s
1439
1443
1440 def _matchstatus(self, other, match):
1444 def _matchstatus(self, other, match):
1441 """override the match method with a filter for directory patterns
1445 """override the match method with a filter for directory patterns
1442
1446
1443 We use inheritance to customize the match.bad method only in cases of
1447 We use inheritance to customize the match.bad method only in cases of
1444 workingctx since it belongs only to the working directory when
1448 workingctx since it belongs only to the working directory when
1445 comparing against the parent changeset.
1449 comparing against the parent changeset.
1446
1450
1447 If we aren't comparing against the working directory's parent, then we
1451 If we aren't comparing against the working directory's parent, then we
1448 just use the default match object sent to us.
1452 just use the default match object sent to us.
1449 """
1453 """
1450 superself = super(workingctx, self)
1454 superself = super(workingctx, self)
1451 match = superself._matchstatus(other, match)
1455 match = superself._matchstatus(other, match)
1452 if other != self._repo['.']:
1456 if other != self._repo['.']:
1453 def bad(f, msg):
1457 def bad(f, msg):
1454 # 'f' may be a directory pattern from 'match.files()',
1458 # 'f' may be a directory pattern from 'match.files()',
1455 # so 'f not in ctx1' is not enough
1459 # so 'f not in ctx1' is not enough
1456 if f not in other and f not in other.dirs():
1460 if f not in other and f not in other.dirs():
1457 self._repo.ui.warn('%s: %s\n' %
1461 self._repo.ui.warn('%s: %s\n' %
1458 (self._repo.dirstate.pathto(f), msg))
1462 (self._repo.dirstate.pathto(f), msg))
1459 match.bad = bad
1463 match.bad = bad
1460 return match
1464 return match
1461
1465
1462 class committablefilectx(basefilectx):
1466 class committablefilectx(basefilectx):
1463 """A committablefilectx provides common functionality for a file context
1467 """A committablefilectx provides common functionality for a file context
1464 that wants the ability to commit, e.g. workingfilectx or memfilectx."""
1468 that wants the ability to commit, e.g. workingfilectx or memfilectx."""
1465 def __init__(self, repo, path, filelog=None, ctx=None):
1469 def __init__(self, repo, path, filelog=None, ctx=None):
1466 self._repo = repo
1470 self._repo = repo
1467 self._path = path
1471 self._path = path
1468 self._changeid = None
1472 self._changeid = None
1469 self._filerev = self._filenode = None
1473 self._filerev = self._filenode = None
1470
1474
1471 if filelog is not None:
1475 if filelog is not None:
1472 self._filelog = filelog
1476 self._filelog = filelog
1473 if ctx:
1477 if ctx:
1474 self._changectx = ctx
1478 self._changectx = ctx
1475
1479
1476 def __nonzero__(self):
1480 def __nonzero__(self):
1477 return True
1481 return True
1478
1482
1479 def parents(self):
1483 def parents(self):
1480 '''return parent filectxs, following copies if necessary'''
1484 '''return parent filectxs, following copies if necessary'''
1481 def filenode(ctx, path):
1485 def filenode(ctx, path):
1482 return ctx._manifest.get(path, nullid)
1486 return ctx._manifest.get(path, nullid)
1483
1487
1484 path = self._path
1488 path = self._path
1485 fl = self._filelog
1489 fl = self._filelog
1486 pcl = self._changectx._parents
1490 pcl = self._changectx._parents
1487 renamed = self.renamed()
1491 renamed = self.renamed()
1488
1492
1489 if renamed:
1493 if renamed:
1490 pl = [renamed + (None,)]
1494 pl = [renamed + (None,)]
1491 else:
1495 else:
1492 pl = [(path, filenode(pcl[0], path), fl)]
1496 pl = [(path, filenode(pcl[0], path), fl)]
1493
1497
1494 for pc in pcl[1:]:
1498 for pc in pcl[1:]:
1495 pl.append((path, filenode(pc, path), fl))
1499 pl.append((path, filenode(pc, path), fl))
1496
1500
1497 return [filectx(self._repo, p, fileid=n, filelog=l)
1501 return [filectx(self._repo, p, fileid=n, filelog=l)
1498 for p, n, l in pl if n != nullid]
1502 for p, n, l in pl if n != nullid]
1499
1503
1500 def children(self):
1504 def children(self):
1501 return []
1505 return []
1502
1506
1503 class workingfilectx(committablefilectx):
1507 class workingfilectx(committablefilectx):
1504 """A workingfilectx object makes access to data related to a particular
1508 """A workingfilectx object makes access to data related to a particular
1505 file in the working directory convenient."""
1509 file in the working directory convenient."""
1506 def __init__(self, repo, path, filelog=None, workingctx=None):
1510 def __init__(self, repo, path, filelog=None, workingctx=None):
1507 super(workingfilectx, self).__init__(repo, path, filelog, workingctx)
1511 super(workingfilectx, self).__init__(repo, path, filelog, workingctx)
1508
1512
1509 @propertycache
1513 @propertycache
1510 def _changectx(self):
1514 def _changectx(self):
1511 return workingctx(self._repo)
1515 return workingctx(self._repo)
1512
1516
1513 def data(self):
1517 def data(self):
1514 return self._repo.wread(self._path)
1518 return self._repo.wread(self._path)
1515 def renamed(self):
1519 def renamed(self):
1516 rp = self._repo.dirstate.copied(self._path)
1520 rp = self._repo.dirstate.copied(self._path)
1517 if not rp:
1521 if not rp:
1518 return None
1522 return None
1519 return rp, self._changectx._parents[0]._manifest.get(rp, nullid)
1523 return rp, self._changectx._parents[0]._manifest.get(rp, nullid)
1520
1524
1521 def size(self):
1525 def size(self):
1522 return self._repo.wvfs.lstat(self._path).st_size
1526 return self._repo.wvfs.lstat(self._path).st_size
1523 def date(self):
1527 def date(self):
1524 t, tz = self._changectx.date()
1528 t, tz = self._changectx.date()
1525 try:
1529 try:
1526 return (int(self._repo.wvfs.lstat(self._path).st_mtime), tz)
1530 return (int(self._repo.wvfs.lstat(self._path).st_mtime), tz)
1527 except OSError, err:
1531 except OSError, err:
1528 if err.errno != errno.ENOENT:
1532 if err.errno != errno.ENOENT:
1529 raise
1533 raise
1530 return (t, tz)
1534 return (t, tz)
1531
1535
1532 def cmp(self, fctx):
1536 def cmp(self, fctx):
1533 """compare with other file context
1537 """compare with other file context
1534
1538
1535 returns True if different than fctx.
1539 returns True if different than fctx.
1536 """
1540 """
1537 # fctx should be a filectx (not a workingfilectx)
1541 # fctx should be a filectx (not a workingfilectx)
1538 # invert comparison to reuse the same code path
1542 # invert comparison to reuse the same code path
1539 return fctx.cmp(self)
1543 return fctx.cmp(self)
1540
1544
1541 def remove(self, ignoremissing=False):
1545 def remove(self, ignoremissing=False):
1542 """wraps unlink for a repo's working directory"""
1546 """wraps unlink for a repo's working directory"""
1543 util.unlinkpath(self._repo.wjoin(self._path), ignoremissing)
1547 util.unlinkpath(self._repo.wjoin(self._path), ignoremissing)
1544
1548
1545 def write(self, data, flags):
1549 def write(self, data, flags):
1546 """wraps repo.wwrite"""
1550 """wraps repo.wwrite"""
1547 self._repo.wwrite(self._path, data, flags)
1551 self._repo.wwrite(self._path, data, flags)
1548
1552
1549 class memctx(committablectx):
1553 class memctx(committablectx):
1550 """Use memctx to perform in-memory commits via localrepo.commitctx().
1554 """Use memctx to perform in-memory commits via localrepo.commitctx().
1551
1555
1552 Revision information is supplied at initialization time while
1556 Revision information is supplied at initialization time while
1553 related files data and is made available through a callback
1557 related files data and is made available through a callback
1554 mechanism. 'repo' is the current localrepo, 'parents' is a
1558 mechanism. 'repo' is the current localrepo, 'parents' is a
1555 sequence of two parent revisions identifiers (pass None for every
1559 sequence of two parent revisions identifiers (pass None for every
1556 missing parent), 'text' is the commit message and 'files' lists
1560 missing parent), 'text' is the commit message and 'files' lists
1557 names of files touched by the revision (normalized and relative to
1561 names of files touched by the revision (normalized and relative to
1558 repository root).
1562 repository root).
1559
1563
1560 filectxfn(repo, memctx, path) is a callable receiving the
1564 filectxfn(repo, memctx, path) is a callable receiving the
1561 repository, the current memctx object and the normalized path of
1565 repository, the current memctx object and the normalized path of
1562 requested file, relative to repository root. It is fired by the
1566 requested file, relative to repository root. It is fired by the
1563 commit function for every file in 'files', but calls order is
1567 commit function for every file in 'files', but calls order is
1564 undefined. If the file is available in the revision being
1568 undefined. If the file is available in the revision being
1565 committed (updated or added), filectxfn returns a memfilectx
1569 committed (updated or added), filectxfn returns a memfilectx
1566 object. If the file was removed, filectxfn raises an
1570 object. If the file was removed, filectxfn raises an
1567 IOError. Moved files are represented by marking the source file
1571 IOError. Moved files are represented by marking the source file
1568 removed and the new file added with copy information (see
1572 removed and the new file added with copy information (see
1569 memfilectx).
1573 memfilectx).
1570
1574
1571 user receives the committer name and defaults to current
1575 user receives the committer name and defaults to current
1572 repository username, date is the commit date in any format
1576 repository username, date is the commit date in any format
1573 supported by util.parsedate() and defaults to current date, extra
1577 supported by util.parsedate() and defaults to current date, extra
1574 is a dictionary of metadata or is left empty.
1578 is a dictionary of metadata or is left empty.
1575 """
1579 """
1576
1580
1577 # Mercurial <= 3.1 expects the filectxfn to raise IOError for missing files.
1581 # Mercurial <= 3.1 expects the filectxfn to raise IOError for missing files.
1578 # Extensions that need to retain compatibility across Mercurial 3.1 can use
1582 # Extensions that need to retain compatibility across Mercurial 3.1 can use
1579 # this field to determine what to do in filectxfn.
1583 # this field to determine what to do in filectxfn.
1580 _returnnoneformissingfiles = True
1584 _returnnoneformissingfiles = True
1581
1585
1582 def __init__(self, repo, parents, text, files, filectxfn, user=None,
1586 def __init__(self, repo, parents, text, files, filectxfn, user=None,
1583 date=None, extra=None, editor=False):
1587 date=None, extra=None, editor=False):
1584 super(memctx, self).__init__(repo, text, user, date, extra)
1588 super(memctx, self).__init__(repo, text, user, date, extra)
1585 self._rev = None
1589 self._rev = None
1586 self._node = None
1590 self._node = None
1587 parents = [(p or nullid) for p in parents]
1591 parents = [(p or nullid) for p in parents]
1588 p1, p2 = parents
1592 p1, p2 = parents
1589 self._parents = [changectx(self._repo, p) for p in (p1, p2)]
1593 self._parents = [changectx(self._repo, p) for p in (p1, p2)]
1590 files = sorted(set(files))
1594 files = sorted(set(files))
1591 self._status = scmutil.status(files, [], [], [], [], [], [])
1595 self._status = scmutil.status(files, [], [], [], [], [], [])
1592 self._filectxfn = filectxfn
1596 self._filectxfn = filectxfn
1593 self.substate = {}
1597 self.substate = {}
1594
1598
1595 # if store is not callable, wrap it in a function
1599 # if store is not callable, wrap it in a function
1596 if not callable(filectxfn):
1600 if not callable(filectxfn):
1597 def getfilectx(repo, memctx, path):
1601 def getfilectx(repo, memctx, path):
1598 fctx = filectxfn[path]
1602 fctx = filectxfn[path]
1599 # this is weird but apparently we only keep track of one parent
1603 # this is weird but apparently we only keep track of one parent
1600 # (why not only store that instead of a tuple?)
1604 # (why not only store that instead of a tuple?)
1601 copied = fctx.renamed()
1605 copied = fctx.renamed()
1602 if copied:
1606 if copied:
1603 copied = copied[0]
1607 copied = copied[0]
1604 return memfilectx(repo, path, fctx.data(),
1608 return memfilectx(repo, path, fctx.data(),
1605 islink=fctx.islink(), isexec=fctx.isexec(),
1609 islink=fctx.islink(), isexec=fctx.isexec(),
1606 copied=copied, memctx=memctx)
1610 copied=copied, memctx=memctx)
1607 self._filectxfn = getfilectx
1611 self._filectxfn = getfilectx
1608
1612
1609 self._extra = extra and extra.copy() or {}
1613 self._extra = extra and extra.copy() or {}
1610 if self._extra.get('branch', '') == '':
1614 if self._extra.get('branch', '') == '':
1611 self._extra['branch'] = 'default'
1615 self._extra['branch'] = 'default'
1612
1616
1613 if editor:
1617 if editor:
1614 self._text = editor(self._repo, self, [])
1618 self._text = editor(self._repo, self, [])
1615 self._repo.savecommitmessage(self._text)
1619 self._repo.savecommitmessage(self._text)
1616
1620
1617 def filectx(self, path, filelog=None):
1621 def filectx(self, path, filelog=None):
1618 """get a file context from the working directory
1622 """get a file context from the working directory
1619
1623
1620 Returns None if file doesn't exist and should be removed."""
1624 Returns None if file doesn't exist and should be removed."""
1621 return self._filectxfn(self._repo, self, path)
1625 return self._filectxfn(self._repo, self, path)
1622
1626
1623 def commit(self):
1627 def commit(self):
1624 """commit context to the repo"""
1628 """commit context to the repo"""
1625 return self._repo.commitctx(self)
1629 return self._repo.commitctx(self)
1626
1630
1627 @propertycache
1631 @propertycache
1628 def _manifest(self):
1632 def _manifest(self):
1629 """generate a manifest based on the return values of filectxfn"""
1633 """generate a manifest based on the return values of filectxfn"""
1630
1634
1631 # keep this simple for now; just worry about p1
1635 # keep this simple for now; just worry about p1
1632 pctx = self._parents[0]
1636 pctx = self._parents[0]
1633 man = pctx.manifest().copy()
1637 man = pctx.manifest().copy()
1634
1638
1635 for f, fnode in man.iteritems():
1639 for f, fnode in man.iteritems():
1636 p1node = nullid
1640 p1node = nullid
1637 p2node = nullid
1641 p2node = nullid
1638 p = pctx[f].parents() # if file isn't in pctx, check p2?
1642 p = pctx[f].parents() # if file isn't in pctx, check p2?
1639 if len(p) > 0:
1643 if len(p) > 0:
1640 p1node = p[0].node()
1644 p1node = p[0].node()
1641 if len(p) > 1:
1645 if len(p) > 1:
1642 p2node = p[1].node()
1646 p2node = p[1].node()
1643 man[f] = revlog.hash(self[f].data(), p1node, p2node)
1647 man[f] = revlog.hash(self[f].data(), p1node, p2node)
1644
1648
1645 return man
1649 return man
1646
1650
1647
1651
1648 class memfilectx(committablefilectx):
1652 class memfilectx(committablefilectx):
1649 """memfilectx represents an in-memory file to commit.
1653 """memfilectx represents an in-memory file to commit.
1650
1654
1651 See memctx and committablefilectx for more details.
1655 See memctx and committablefilectx for more details.
1652 """
1656 """
1653 def __init__(self, repo, path, data, islink=False,
1657 def __init__(self, repo, path, data, islink=False,
1654 isexec=False, copied=None, memctx=None):
1658 isexec=False, copied=None, memctx=None):
1655 """
1659 """
1656 path is the normalized file path relative to repository root.
1660 path is the normalized file path relative to repository root.
1657 data is the file content as a string.
1661 data is the file content as a string.
1658 islink is True if the file is a symbolic link.
1662 islink is True if the file is a symbolic link.
1659 isexec is True if the file is executable.
1663 isexec is True if the file is executable.
1660 copied is the source file path if current file was copied in the
1664 copied is the source file path if current file was copied in the
1661 revision being committed, or None."""
1665 revision being committed, or None."""
1662 super(memfilectx, self).__init__(repo, path, None, memctx)
1666 super(memfilectx, self).__init__(repo, path, None, memctx)
1663 self._data = data
1667 self._data = data
1664 self._flags = (islink and 'l' or '') + (isexec and 'x' or '')
1668 self._flags = (islink and 'l' or '') + (isexec and 'x' or '')
1665 self._copied = None
1669 self._copied = None
1666 if copied:
1670 if copied:
1667 self._copied = (copied, nullid)
1671 self._copied = (copied, nullid)
1668
1672
1669 def data(self):
1673 def data(self):
1670 return self._data
1674 return self._data
1671 def size(self):
1675 def size(self):
1672 return len(self.data())
1676 return len(self.data())
1673 def flags(self):
1677 def flags(self):
1674 return self._flags
1678 return self._flags
1675 def renamed(self):
1679 def renamed(self):
1676 return self._copied
1680 return self._copied
1677
1681
1678 def remove(self, ignoremissing=False):
1682 def remove(self, ignoremissing=False):
1679 """wraps unlink for a repo's working directory"""
1683 """wraps unlink for a repo's working directory"""
1680 # need to figure out what to do here
1684 # need to figure out what to do here
1681 del self._changectx[self._path]
1685 del self._changectx[self._path]
1682
1686
1683 def write(self, data, flags):
1687 def write(self, data, flags):
1684 """wraps repo.wwrite"""
1688 """wraps repo.wwrite"""
1685 self._data = data
1689 self._data = data
General Comments 0
You need to be logged in to leave comments. Login now