##// END OF EJS Templates
revset: fix crash on empty sort key...
Yuya Nishihara -
r29362:ec75d77d default
parent child Browse files
Show More
@@ -1,3663 +1,3665
1 # revset.py - revision set queries for mercurial
1 # revset.py - revision set queries for mercurial
2 #
2 #
3 # Copyright 2010 Matt Mackall <mpm@selenic.com>
3 # Copyright 2010 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 __future__ import absolute_import
8 from __future__ import absolute_import
9
9
10 import heapq
10 import heapq
11 import re
11 import re
12
12
13 from .i18n import _
13 from .i18n import _
14 from . import (
14 from . import (
15 destutil,
15 destutil,
16 encoding,
16 encoding,
17 error,
17 error,
18 hbisect,
18 hbisect,
19 match as matchmod,
19 match as matchmod,
20 node,
20 node,
21 obsolete as obsmod,
21 obsolete as obsmod,
22 parser,
22 parser,
23 pathutil,
23 pathutil,
24 phases,
24 phases,
25 registrar,
25 registrar,
26 repoview,
26 repoview,
27 util,
27 util,
28 )
28 )
29
29
30 def _revancestors(repo, revs, followfirst):
30 def _revancestors(repo, revs, followfirst):
31 """Like revlog.ancestors(), but supports followfirst."""
31 """Like revlog.ancestors(), but supports followfirst."""
32 if followfirst:
32 if followfirst:
33 cut = 1
33 cut = 1
34 else:
34 else:
35 cut = None
35 cut = None
36 cl = repo.changelog
36 cl = repo.changelog
37
37
38 def iterate():
38 def iterate():
39 revs.sort(reverse=True)
39 revs.sort(reverse=True)
40 irevs = iter(revs)
40 irevs = iter(revs)
41 h = []
41 h = []
42
42
43 inputrev = next(irevs, None)
43 inputrev = next(irevs, None)
44 if inputrev is not None:
44 if inputrev is not None:
45 heapq.heappush(h, -inputrev)
45 heapq.heappush(h, -inputrev)
46
46
47 seen = set()
47 seen = set()
48 while h:
48 while h:
49 current = -heapq.heappop(h)
49 current = -heapq.heappop(h)
50 if current == inputrev:
50 if current == inputrev:
51 inputrev = next(irevs, None)
51 inputrev = next(irevs, None)
52 if inputrev is not None:
52 if inputrev is not None:
53 heapq.heappush(h, -inputrev)
53 heapq.heappush(h, -inputrev)
54 if current not in seen:
54 if current not in seen:
55 seen.add(current)
55 seen.add(current)
56 yield current
56 yield current
57 for parent in cl.parentrevs(current)[:cut]:
57 for parent in cl.parentrevs(current)[:cut]:
58 if parent != node.nullrev:
58 if parent != node.nullrev:
59 heapq.heappush(h, -parent)
59 heapq.heappush(h, -parent)
60
60
61 return generatorset(iterate(), iterasc=False)
61 return generatorset(iterate(), iterasc=False)
62
62
63 def _revdescendants(repo, revs, followfirst):
63 def _revdescendants(repo, revs, followfirst):
64 """Like revlog.descendants() but supports followfirst."""
64 """Like revlog.descendants() but supports followfirst."""
65 if followfirst:
65 if followfirst:
66 cut = 1
66 cut = 1
67 else:
67 else:
68 cut = None
68 cut = None
69
69
70 def iterate():
70 def iterate():
71 cl = repo.changelog
71 cl = repo.changelog
72 # XXX this should be 'parentset.min()' assuming 'parentset' is a
72 # XXX this should be 'parentset.min()' assuming 'parentset' is a
73 # smartset (and if it is not, it should.)
73 # smartset (and if it is not, it should.)
74 first = min(revs)
74 first = min(revs)
75 nullrev = node.nullrev
75 nullrev = node.nullrev
76 if first == nullrev:
76 if first == nullrev:
77 # Are there nodes with a null first parent and a non-null
77 # Are there nodes with a null first parent and a non-null
78 # second one? Maybe. Do we care? Probably not.
78 # second one? Maybe. Do we care? Probably not.
79 for i in cl:
79 for i in cl:
80 yield i
80 yield i
81 else:
81 else:
82 seen = set(revs)
82 seen = set(revs)
83 for i in cl.revs(first + 1):
83 for i in cl.revs(first + 1):
84 for x in cl.parentrevs(i)[:cut]:
84 for x in cl.parentrevs(i)[:cut]:
85 if x != nullrev and x in seen:
85 if x != nullrev and x in seen:
86 seen.add(i)
86 seen.add(i)
87 yield i
87 yield i
88 break
88 break
89
89
90 return generatorset(iterate(), iterasc=True)
90 return generatorset(iterate(), iterasc=True)
91
91
92 def _reachablerootspure(repo, minroot, roots, heads, includepath):
92 def _reachablerootspure(repo, minroot, roots, heads, includepath):
93 """return (heads(::<roots> and ::<heads>))
93 """return (heads(::<roots> and ::<heads>))
94
94
95 If includepath is True, return (<roots>::<heads>)."""
95 If includepath is True, return (<roots>::<heads>)."""
96 if not roots:
96 if not roots:
97 return []
97 return []
98 parentrevs = repo.changelog.parentrevs
98 parentrevs = repo.changelog.parentrevs
99 roots = set(roots)
99 roots = set(roots)
100 visit = list(heads)
100 visit = list(heads)
101 reachable = set()
101 reachable = set()
102 seen = {}
102 seen = {}
103 # prefetch all the things! (because python is slow)
103 # prefetch all the things! (because python is slow)
104 reached = reachable.add
104 reached = reachable.add
105 dovisit = visit.append
105 dovisit = visit.append
106 nextvisit = visit.pop
106 nextvisit = visit.pop
107 # open-code the post-order traversal due to the tiny size of
107 # open-code the post-order traversal due to the tiny size of
108 # sys.getrecursionlimit()
108 # sys.getrecursionlimit()
109 while visit:
109 while visit:
110 rev = nextvisit()
110 rev = nextvisit()
111 if rev in roots:
111 if rev in roots:
112 reached(rev)
112 reached(rev)
113 if not includepath:
113 if not includepath:
114 continue
114 continue
115 parents = parentrevs(rev)
115 parents = parentrevs(rev)
116 seen[rev] = parents
116 seen[rev] = parents
117 for parent in parents:
117 for parent in parents:
118 if parent >= minroot and parent not in seen:
118 if parent >= minroot and parent not in seen:
119 dovisit(parent)
119 dovisit(parent)
120 if not reachable:
120 if not reachable:
121 return baseset()
121 return baseset()
122 if not includepath:
122 if not includepath:
123 return reachable
123 return reachable
124 for rev in sorted(seen):
124 for rev in sorted(seen):
125 for parent in seen[rev]:
125 for parent in seen[rev]:
126 if parent in reachable:
126 if parent in reachable:
127 reached(rev)
127 reached(rev)
128 return reachable
128 return reachable
129
129
130 def reachableroots(repo, roots, heads, includepath=False):
130 def reachableroots(repo, roots, heads, includepath=False):
131 """return (heads(::<roots> and ::<heads>))
131 """return (heads(::<roots> and ::<heads>))
132
132
133 If includepath is True, return (<roots>::<heads>)."""
133 If includepath is True, return (<roots>::<heads>)."""
134 if not roots:
134 if not roots:
135 return baseset()
135 return baseset()
136 minroot = roots.min()
136 minroot = roots.min()
137 roots = list(roots)
137 roots = list(roots)
138 heads = list(heads)
138 heads = list(heads)
139 try:
139 try:
140 revs = repo.changelog.reachableroots(minroot, heads, roots, includepath)
140 revs = repo.changelog.reachableroots(minroot, heads, roots, includepath)
141 except AttributeError:
141 except AttributeError:
142 revs = _reachablerootspure(repo, minroot, roots, heads, includepath)
142 revs = _reachablerootspure(repo, minroot, roots, heads, includepath)
143 revs = baseset(revs)
143 revs = baseset(revs)
144 revs.sort()
144 revs.sort()
145 return revs
145 return revs
146
146
147 elements = {
147 elements = {
148 # token-type: binding-strength, primary, prefix, infix, suffix
148 # token-type: binding-strength, primary, prefix, infix, suffix
149 "(": (21, None, ("group", 1, ")"), ("func", 1, ")"), None),
149 "(": (21, None, ("group", 1, ")"), ("func", 1, ")"), None),
150 "##": (20, None, None, ("_concat", 20), None),
150 "##": (20, None, None, ("_concat", 20), None),
151 "~": (18, None, None, ("ancestor", 18), None),
151 "~": (18, None, None, ("ancestor", 18), None),
152 "^": (18, None, None, ("parent", 18), ("parentpost", 18)),
152 "^": (18, None, None, ("parent", 18), ("parentpost", 18)),
153 "-": (5, None, ("negate", 19), ("minus", 5), None),
153 "-": (5, None, ("negate", 19), ("minus", 5), None),
154 "::": (17, None, ("dagrangepre", 17), ("dagrange", 17),
154 "::": (17, None, ("dagrangepre", 17), ("dagrange", 17),
155 ("dagrangepost", 17)),
155 ("dagrangepost", 17)),
156 "..": (17, None, ("dagrangepre", 17), ("dagrange", 17),
156 "..": (17, None, ("dagrangepre", 17), ("dagrange", 17),
157 ("dagrangepost", 17)),
157 ("dagrangepost", 17)),
158 ":": (15, "rangeall", ("rangepre", 15), ("range", 15), ("rangepost", 15)),
158 ":": (15, "rangeall", ("rangepre", 15), ("range", 15), ("rangepost", 15)),
159 "not": (10, None, ("not", 10), None, None),
159 "not": (10, None, ("not", 10), None, None),
160 "!": (10, None, ("not", 10), None, None),
160 "!": (10, None, ("not", 10), None, None),
161 "and": (5, None, None, ("and", 5), None),
161 "and": (5, None, None, ("and", 5), None),
162 "&": (5, None, None, ("and", 5), None),
162 "&": (5, None, None, ("and", 5), None),
163 "%": (5, None, None, ("only", 5), ("onlypost", 5)),
163 "%": (5, None, None, ("only", 5), ("onlypost", 5)),
164 "or": (4, None, None, ("or", 4), None),
164 "or": (4, None, None, ("or", 4), None),
165 "|": (4, None, None, ("or", 4), None),
165 "|": (4, None, None, ("or", 4), None),
166 "+": (4, None, None, ("or", 4), None),
166 "+": (4, None, None, ("or", 4), None),
167 "=": (3, None, None, ("keyvalue", 3), None),
167 "=": (3, None, None, ("keyvalue", 3), None),
168 ",": (2, None, None, ("list", 2), None),
168 ",": (2, None, None, ("list", 2), None),
169 ")": (0, None, None, None, None),
169 ")": (0, None, None, None, None),
170 "symbol": (0, "symbol", None, None, None),
170 "symbol": (0, "symbol", None, None, None),
171 "string": (0, "string", None, None, None),
171 "string": (0, "string", None, None, None),
172 "end": (0, None, None, None, None),
172 "end": (0, None, None, None, None),
173 }
173 }
174
174
175 keywords = set(['and', 'or', 'not'])
175 keywords = set(['and', 'or', 'not'])
176
176
177 # default set of valid characters for the initial letter of symbols
177 # default set of valid characters for the initial letter of symbols
178 _syminitletters = set(c for c in [chr(i) for i in xrange(256)]
178 _syminitletters = set(c for c in [chr(i) for i in xrange(256)]
179 if c.isalnum() or c in '._@' or ord(c) > 127)
179 if c.isalnum() or c in '._@' or ord(c) > 127)
180
180
181 # default set of valid characters for non-initial letters of symbols
181 # default set of valid characters for non-initial letters of symbols
182 _symletters = set(c for c in [chr(i) for i in xrange(256)]
182 _symletters = set(c for c in [chr(i) for i in xrange(256)]
183 if c.isalnum() or c in '-._/@' or ord(c) > 127)
183 if c.isalnum() or c in '-._/@' or ord(c) > 127)
184
184
185 def tokenize(program, lookup=None, syminitletters=None, symletters=None):
185 def tokenize(program, lookup=None, syminitletters=None, symletters=None):
186 '''
186 '''
187 Parse a revset statement into a stream of tokens
187 Parse a revset statement into a stream of tokens
188
188
189 ``syminitletters`` is the set of valid characters for the initial
189 ``syminitletters`` is the set of valid characters for the initial
190 letter of symbols.
190 letter of symbols.
191
191
192 By default, character ``c`` is recognized as valid for initial
192 By default, character ``c`` is recognized as valid for initial
193 letter of symbols, if ``c.isalnum() or c in '._@' or ord(c) > 127``.
193 letter of symbols, if ``c.isalnum() or c in '._@' or ord(c) > 127``.
194
194
195 ``symletters`` is the set of valid characters for non-initial
195 ``symletters`` is the set of valid characters for non-initial
196 letters of symbols.
196 letters of symbols.
197
197
198 By default, character ``c`` is recognized as valid for non-initial
198 By default, character ``c`` is recognized as valid for non-initial
199 letters of symbols, if ``c.isalnum() or c in '-._/@' or ord(c) > 127``.
199 letters of symbols, if ``c.isalnum() or c in '-._/@' or ord(c) > 127``.
200
200
201 Check that @ is a valid unquoted token character (issue3686):
201 Check that @ is a valid unquoted token character (issue3686):
202 >>> list(tokenize("@::"))
202 >>> list(tokenize("@::"))
203 [('symbol', '@', 0), ('::', None, 1), ('end', None, 3)]
203 [('symbol', '@', 0), ('::', None, 1), ('end', None, 3)]
204
204
205 '''
205 '''
206 if syminitletters is None:
206 if syminitletters is None:
207 syminitletters = _syminitletters
207 syminitletters = _syminitletters
208 if symletters is None:
208 if symletters is None:
209 symletters = _symletters
209 symletters = _symletters
210
210
211 if program and lookup:
211 if program and lookup:
212 # attempt to parse old-style ranges first to deal with
212 # attempt to parse old-style ranges first to deal with
213 # things like old-tag which contain query metacharacters
213 # things like old-tag which contain query metacharacters
214 parts = program.split(':', 1)
214 parts = program.split(':', 1)
215 if all(lookup(sym) for sym in parts if sym):
215 if all(lookup(sym) for sym in parts if sym):
216 if parts[0]:
216 if parts[0]:
217 yield ('symbol', parts[0], 0)
217 yield ('symbol', parts[0], 0)
218 if len(parts) > 1:
218 if len(parts) > 1:
219 s = len(parts[0])
219 s = len(parts[0])
220 yield (':', None, s)
220 yield (':', None, s)
221 if parts[1]:
221 if parts[1]:
222 yield ('symbol', parts[1], s + 1)
222 yield ('symbol', parts[1], s + 1)
223 yield ('end', None, len(program))
223 yield ('end', None, len(program))
224 return
224 return
225
225
226 pos, l = 0, len(program)
226 pos, l = 0, len(program)
227 while pos < l:
227 while pos < l:
228 c = program[pos]
228 c = program[pos]
229 if c.isspace(): # skip inter-token whitespace
229 if c.isspace(): # skip inter-token whitespace
230 pass
230 pass
231 elif c == ':' and program[pos:pos + 2] == '::': # look ahead carefully
231 elif c == ':' and program[pos:pos + 2] == '::': # look ahead carefully
232 yield ('::', None, pos)
232 yield ('::', None, pos)
233 pos += 1 # skip ahead
233 pos += 1 # skip ahead
234 elif c == '.' and program[pos:pos + 2] == '..': # look ahead carefully
234 elif c == '.' and program[pos:pos + 2] == '..': # look ahead carefully
235 yield ('..', None, pos)
235 yield ('..', None, pos)
236 pos += 1 # skip ahead
236 pos += 1 # skip ahead
237 elif c == '#' and program[pos:pos + 2] == '##': # look ahead carefully
237 elif c == '#' and program[pos:pos + 2] == '##': # look ahead carefully
238 yield ('##', None, pos)
238 yield ('##', None, pos)
239 pos += 1 # skip ahead
239 pos += 1 # skip ahead
240 elif c in "():=,-|&+!~^%": # handle simple operators
240 elif c in "():=,-|&+!~^%": # handle simple operators
241 yield (c, None, pos)
241 yield (c, None, pos)
242 elif (c in '"\'' or c == 'r' and
242 elif (c in '"\'' or c == 'r' and
243 program[pos:pos + 2] in ("r'", 'r"')): # handle quoted strings
243 program[pos:pos + 2] in ("r'", 'r"')): # handle quoted strings
244 if c == 'r':
244 if c == 'r':
245 pos += 1
245 pos += 1
246 c = program[pos]
246 c = program[pos]
247 decode = lambda x: x
247 decode = lambda x: x
248 else:
248 else:
249 decode = parser.unescapestr
249 decode = parser.unescapestr
250 pos += 1
250 pos += 1
251 s = pos
251 s = pos
252 while pos < l: # find closing quote
252 while pos < l: # find closing quote
253 d = program[pos]
253 d = program[pos]
254 if d == '\\': # skip over escaped characters
254 if d == '\\': # skip over escaped characters
255 pos += 2
255 pos += 2
256 continue
256 continue
257 if d == c:
257 if d == c:
258 yield ('string', decode(program[s:pos]), s)
258 yield ('string', decode(program[s:pos]), s)
259 break
259 break
260 pos += 1
260 pos += 1
261 else:
261 else:
262 raise error.ParseError(_("unterminated string"), s)
262 raise error.ParseError(_("unterminated string"), s)
263 # gather up a symbol/keyword
263 # gather up a symbol/keyword
264 elif c in syminitletters:
264 elif c in syminitletters:
265 s = pos
265 s = pos
266 pos += 1
266 pos += 1
267 while pos < l: # find end of symbol
267 while pos < l: # find end of symbol
268 d = program[pos]
268 d = program[pos]
269 if d not in symletters:
269 if d not in symletters:
270 break
270 break
271 if d == '.' and program[pos - 1] == '.': # special case for ..
271 if d == '.' and program[pos - 1] == '.': # special case for ..
272 pos -= 1
272 pos -= 1
273 break
273 break
274 pos += 1
274 pos += 1
275 sym = program[s:pos]
275 sym = program[s:pos]
276 if sym in keywords: # operator keywords
276 if sym in keywords: # operator keywords
277 yield (sym, None, s)
277 yield (sym, None, s)
278 elif '-' in sym:
278 elif '-' in sym:
279 # some jerk gave us foo-bar-baz, try to check if it's a symbol
279 # some jerk gave us foo-bar-baz, try to check if it's a symbol
280 if lookup and lookup(sym):
280 if lookup and lookup(sym):
281 # looks like a real symbol
281 # looks like a real symbol
282 yield ('symbol', sym, s)
282 yield ('symbol', sym, s)
283 else:
283 else:
284 # looks like an expression
284 # looks like an expression
285 parts = sym.split('-')
285 parts = sym.split('-')
286 for p in parts[:-1]:
286 for p in parts[:-1]:
287 if p: # possible consecutive -
287 if p: # possible consecutive -
288 yield ('symbol', p, s)
288 yield ('symbol', p, s)
289 s += len(p)
289 s += len(p)
290 yield ('-', None, pos)
290 yield ('-', None, pos)
291 s += 1
291 s += 1
292 if parts[-1]: # possible trailing -
292 if parts[-1]: # possible trailing -
293 yield ('symbol', parts[-1], s)
293 yield ('symbol', parts[-1], s)
294 else:
294 else:
295 yield ('symbol', sym, s)
295 yield ('symbol', sym, s)
296 pos -= 1
296 pos -= 1
297 else:
297 else:
298 raise error.ParseError(_("syntax error in revset '%s'") %
298 raise error.ParseError(_("syntax error in revset '%s'") %
299 program, pos)
299 program, pos)
300 pos += 1
300 pos += 1
301 yield ('end', None, pos)
301 yield ('end', None, pos)
302
302
303 # helpers
303 # helpers
304
304
305 def getstring(x, err):
305 def getstring(x, err):
306 if x and (x[0] == 'string' or x[0] == 'symbol'):
306 if x and (x[0] == 'string' or x[0] == 'symbol'):
307 return x[1]
307 return x[1]
308 raise error.ParseError(err)
308 raise error.ParseError(err)
309
309
310 def getlist(x):
310 def getlist(x):
311 if not x:
311 if not x:
312 return []
312 return []
313 if x[0] == 'list':
313 if x[0] == 'list':
314 return list(x[1:])
314 return list(x[1:])
315 return [x]
315 return [x]
316
316
317 def getargs(x, min, max, err):
317 def getargs(x, min, max, err):
318 l = getlist(x)
318 l = getlist(x)
319 if len(l) < min or (max >= 0 and len(l) > max):
319 if len(l) < min or (max >= 0 and len(l) > max):
320 raise error.ParseError(err)
320 raise error.ParseError(err)
321 return l
321 return l
322
322
323 def getargsdict(x, funcname, keys):
323 def getargsdict(x, funcname, keys):
324 return parser.buildargsdict(getlist(x), funcname, keys.split(),
324 return parser.buildargsdict(getlist(x), funcname, keys.split(),
325 keyvaluenode='keyvalue', keynode='symbol')
325 keyvaluenode='keyvalue', keynode='symbol')
326
326
327 def getset(repo, subset, x):
327 def getset(repo, subset, x):
328 if not x:
328 if not x:
329 raise error.ParseError(_("missing argument"))
329 raise error.ParseError(_("missing argument"))
330 s = methods[x[0]](repo, subset, *x[1:])
330 s = methods[x[0]](repo, subset, *x[1:])
331 if util.safehasattr(s, 'isascending'):
331 if util.safehasattr(s, 'isascending'):
332 return s
332 return s
333 # else case should not happen, because all non-func are internal,
333 # else case should not happen, because all non-func are internal,
334 # ignoring for now.
334 # ignoring for now.
335 if x[0] == 'func' and x[1][0] == 'symbol' and x[1][1] in symbols:
335 if x[0] == 'func' and x[1][0] == 'symbol' and x[1][1] in symbols:
336 repo.ui.deprecwarn('revset "%s" uses list instead of smartset'
336 repo.ui.deprecwarn('revset "%s" uses list instead of smartset'
337 % x[1][1],
337 % x[1][1],
338 '3.9')
338 '3.9')
339 return baseset(s)
339 return baseset(s)
340
340
341 def _getrevsource(repo, r):
341 def _getrevsource(repo, r):
342 extra = repo[r].extra()
342 extra = repo[r].extra()
343 for label in ('source', 'transplant_source', 'rebase_source'):
343 for label in ('source', 'transplant_source', 'rebase_source'):
344 if label in extra:
344 if label in extra:
345 try:
345 try:
346 return repo[extra[label]].rev()
346 return repo[extra[label]].rev()
347 except error.RepoLookupError:
347 except error.RepoLookupError:
348 pass
348 pass
349 return None
349 return None
350
350
351 # operator methods
351 # operator methods
352
352
353 def stringset(repo, subset, x):
353 def stringset(repo, subset, x):
354 x = repo[x].rev()
354 x = repo[x].rev()
355 if (x in subset
355 if (x in subset
356 or x == node.nullrev and isinstance(subset, fullreposet)):
356 or x == node.nullrev and isinstance(subset, fullreposet)):
357 return baseset([x])
357 return baseset([x])
358 return baseset()
358 return baseset()
359
359
360 def rangeset(repo, subset, x, y):
360 def rangeset(repo, subset, x, y):
361 m = getset(repo, fullreposet(repo), x)
361 m = getset(repo, fullreposet(repo), x)
362 n = getset(repo, fullreposet(repo), y)
362 n = getset(repo, fullreposet(repo), y)
363
363
364 if not m or not n:
364 if not m or not n:
365 return baseset()
365 return baseset()
366 m, n = m.first(), n.last()
366 m, n = m.first(), n.last()
367
367
368 if m == n:
368 if m == n:
369 r = baseset([m])
369 r = baseset([m])
370 elif n == node.wdirrev:
370 elif n == node.wdirrev:
371 r = spanset(repo, m, len(repo)) + baseset([n])
371 r = spanset(repo, m, len(repo)) + baseset([n])
372 elif m == node.wdirrev:
372 elif m == node.wdirrev:
373 r = baseset([m]) + spanset(repo, len(repo) - 1, n - 1)
373 r = baseset([m]) + spanset(repo, len(repo) - 1, n - 1)
374 elif m < n:
374 elif m < n:
375 r = spanset(repo, m, n + 1)
375 r = spanset(repo, m, n + 1)
376 else:
376 else:
377 r = spanset(repo, m, n - 1)
377 r = spanset(repo, m, n - 1)
378 # XXX We should combine with subset first: 'subset & baseset(...)'. This is
378 # XXX We should combine with subset first: 'subset & baseset(...)'. This is
379 # necessary to ensure we preserve the order in subset.
379 # necessary to ensure we preserve the order in subset.
380 #
380 #
381 # This has performance implication, carrying the sorting over when possible
381 # This has performance implication, carrying the sorting over when possible
382 # would be more efficient.
382 # would be more efficient.
383 return r & subset
383 return r & subset
384
384
385 def dagrange(repo, subset, x, y):
385 def dagrange(repo, subset, x, y):
386 r = fullreposet(repo)
386 r = fullreposet(repo)
387 xs = reachableroots(repo, getset(repo, r, x), getset(repo, r, y),
387 xs = reachableroots(repo, getset(repo, r, x), getset(repo, r, y),
388 includepath=True)
388 includepath=True)
389 return subset & xs
389 return subset & xs
390
390
391 def andset(repo, subset, x, y):
391 def andset(repo, subset, x, y):
392 return getset(repo, getset(repo, subset, x), y)
392 return getset(repo, getset(repo, subset, x), y)
393
393
394 def differenceset(repo, subset, x, y):
394 def differenceset(repo, subset, x, y):
395 return getset(repo, subset, x) - getset(repo, subset, y)
395 return getset(repo, subset, x) - getset(repo, subset, y)
396
396
397 def orset(repo, subset, *xs):
397 def orset(repo, subset, *xs):
398 assert xs
398 assert xs
399 if len(xs) == 1:
399 if len(xs) == 1:
400 return getset(repo, subset, xs[0])
400 return getset(repo, subset, xs[0])
401 p = len(xs) // 2
401 p = len(xs) // 2
402 a = orset(repo, subset, *xs[:p])
402 a = orset(repo, subset, *xs[:p])
403 b = orset(repo, subset, *xs[p:])
403 b = orset(repo, subset, *xs[p:])
404 return a + b
404 return a + b
405
405
406 def notset(repo, subset, x):
406 def notset(repo, subset, x):
407 return subset - getset(repo, subset, x)
407 return subset - getset(repo, subset, x)
408
408
409 def listset(repo, subset, *xs):
409 def listset(repo, subset, *xs):
410 raise error.ParseError(_("can't use a list in this context"),
410 raise error.ParseError(_("can't use a list in this context"),
411 hint=_('see hg help "revsets.x or y"'))
411 hint=_('see hg help "revsets.x or y"'))
412
412
413 def keyvaluepair(repo, subset, k, v):
413 def keyvaluepair(repo, subset, k, v):
414 raise error.ParseError(_("can't use a key-value pair in this context"))
414 raise error.ParseError(_("can't use a key-value pair in this context"))
415
415
416 def func(repo, subset, a, b):
416 def func(repo, subset, a, b):
417 if a[0] == 'symbol' and a[1] in symbols:
417 if a[0] == 'symbol' and a[1] in symbols:
418 return symbols[a[1]](repo, subset, b)
418 return symbols[a[1]](repo, subset, b)
419
419
420 keep = lambda fn: getattr(fn, '__doc__', None) is not None
420 keep = lambda fn: getattr(fn, '__doc__', None) is not None
421
421
422 syms = [s for (s, fn) in symbols.items() if keep(fn)]
422 syms = [s for (s, fn) in symbols.items() if keep(fn)]
423 raise error.UnknownIdentifier(a[1], syms)
423 raise error.UnknownIdentifier(a[1], syms)
424
424
425 # functions
425 # functions
426
426
427 # symbols are callables like:
427 # symbols are callables like:
428 # fn(repo, subset, x)
428 # fn(repo, subset, x)
429 # with:
429 # with:
430 # repo - current repository instance
430 # repo - current repository instance
431 # subset - of revisions to be examined
431 # subset - of revisions to be examined
432 # x - argument in tree form
432 # x - argument in tree form
433 symbols = {}
433 symbols = {}
434
434
435 # symbols which can't be used for a DoS attack for any given input
435 # symbols which can't be used for a DoS attack for any given input
436 # (e.g. those which accept regexes as plain strings shouldn't be included)
436 # (e.g. those which accept regexes as plain strings shouldn't be included)
437 # functions that just return a lot of changesets (like all) don't count here
437 # functions that just return a lot of changesets (like all) don't count here
438 safesymbols = set()
438 safesymbols = set()
439
439
440 predicate = registrar.revsetpredicate()
440 predicate = registrar.revsetpredicate()
441
441
442 @predicate('_destupdate')
442 @predicate('_destupdate')
443 def _destupdate(repo, subset, x):
443 def _destupdate(repo, subset, x):
444 # experimental revset for update destination
444 # experimental revset for update destination
445 args = getargsdict(x, 'limit', 'clean check')
445 args = getargsdict(x, 'limit', 'clean check')
446 return subset & baseset([destutil.destupdate(repo, **args)[0]])
446 return subset & baseset([destutil.destupdate(repo, **args)[0]])
447
447
448 @predicate('_destmerge')
448 @predicate('_destmerge')
449 def _destmerge(repo, subset, x):
449 def _destmerge(repo, subset, x):
450 # experimental revset for merge destination
450 # experimental revset for merge destination
451 sourceset = None
451 sourceset = None
452 if x is not None:
452 if x is not None:
453 sourceset = getset(repo, fullreposet(repo), x)
453 sourceset = getset(repo, fullreposet(repo), x)
454 return subset & baseset([destutil.destmerge(repo, sourceset=sourceset)])
454 return subset & baseset([destutil.destmerge(repo, sourceset=sourceset)])
455
455
456 @predicate('adds(pattern)', safe=True)
456 @predicate('adds(pattern)', safe=True)
457 def adds(repo, subset, x):
457 def adds(repo, subset, x):
458 """Changesets that add a file matching pattern.
458 """Changesets that add a file matching pattern.
459
459
460 The pattern without explicit kind like ``glob:`` is expected to be
460 The pattern without explicit kind like ``glob:`` is expected to be
461 relative to the current directory and match against a file or a
461 relative to the current directory and match against a file or a
462 directory.
462 directory.
463 """
463 """
464 # i18n: "adds" is a keyword
464 # i18n: "adds" is a keyword
465 pat = getstring(x, _("adds requires a pattern"))
465 pat = getstring(x, _("adds requires a pattern"))
466 return checkstatus(repo, subset, pat, 1)
466 return checkstatus(repo, subset, pat, 1)
467
467
468 @predicate('ancestor(*changeset)', safe=True)
468 @predicate('ancestor(*changeset)', safe=True)
469 def ancestor(repo, subset, x):
469 def ancestor(repo, subset, x):
470 """A greatest common ancestor of the changesets.
470 """A greatest common ancestor of the changesets.
471
471
472 Accepts 0 or more changesets.
472 Accepts 0 or more changesets.
473 Will return empty list when passed no args.
473 Will return empty list when passed no args.
474 Greatest common ancestor of a single changeset is that changeset.
474 Greatest common ancestor of a single changeset is that changeset.
475 """
475 """
476 # i18n: "ancestor" is a keyword
476 # i18n: "ancestor" is a keyword
477 l = getlist(x)
477 l = getlist(x)
478 rl = fullreposet(repo)
478 rl = fullreposet(repo)
479 anc = None
479 anc = None
480
480
481 # (getset(repo, rl, i) for i in l) generates a list of lists
481 # (getset(repo, rl, i) for i in l) generates a list of lists
482 for revs in (getset(repo, rl, i) for i in l):
482 for revs in (getset(repo, rl, i) for i in l):
483 for r in revs:
483 for r in revs:
484 if anc is None:
484 if anc is None:
485 anc = repo[r]
485 anc = repo[r]
486 else:
486 else:
487 anc = anc.ancestor(repo[r])
487 anc = anc.ancestor(repo[r])
488
488
489 if anc is not None and anc.rev() in subset:
489 if anc is not None and anc.rev() in subset:
490 return baseset([anc.rev()])
490 return baseset([anc.rev()])
491 return baseset()
491 return baseset()
492
492
493 def _ancestors(repo, subset, x, followfirst=False):
493 def _ancestors(repo, subset, x, followfirst=False):
494 heads = getset(repo, fullreposet(repo), x)
494 heads = getset(repo, fullreposet(repo), x)
495 if not heads:
495 if not heads:
496 return baseset()
496 return baseset()
497 s = _revancestors(repo, heads, followfirst)
497 s = _revancestors(repo, heads, followfirst)
498 return subset & s
498 return subset & s
499
499
500 @predicate('ancestors(set)', safe=True)
500 @predicate('ancestors(set)', safe=True)
501 def ancestors(repo, subset, x):
501 def ancestors(repo, subset, x):
502 """Changesets that are ancestors of a changeset in set.
502 """Changesets that are ancestors of a changeset in set.
503 """
503 """
504 return _ancestors(repo, subset, x)
504 return _ancestors(repo, subset, x)
505
505
506 @predicate('_firstancestors', safe=True)
506 @predicate('_firstancestors', safe=True)
507 def _firstancestors(repo, subset, x):
507 def _firstancestors(repo, subset, x):
508 # ``_firstancestors(set)``
508 # ``_firstancestors(set)``
509 # Like ``ancestors(set)`` but follows only the first parents.
509 # Like ``ancestors(set)`` but follows only the first parents.
510 return _ancestors(repo, subset, x, followfirst=True)
510 return _ancestors(repo, subset, x, followfirst=True)
511
511
512 def ancestorspec(repo, subset, x, n):
512 def ancestorspec(repo, subset, x, n):
513 """``set~n``
513 """``set~n``
514 Changesets that are the Nth ancestor (first parents only) of a changeset
514 Changesets that are the Nth ancestor (first parents only) of a changeset
515 in set.
515 in set.
516 """
516 """
517 try:
517 try:
518 n = int(n[1])
518 n = int(n[1])
519 except (TypeError, ValueError):
519 except (TypeError, ValueError):
520 raise error.ParseError(_("~ expects a number"))
520 raise error.ParseError(_("~ expects a number"))
521 ps = set()
521 ps = set()
522 cl = repo.changelog
522 cl = repo.changelog
523 for r in getset(repo, fullreposet(repo), x):
523 for r in getset(repo, fullreposet(repo), x):
524 for i in range(n):
524 for i in range(n):
525 r = cl.parentrevs(r)[0]
525 r = cl.parentrevs(r)[0]
526 ps.add(r)
526 ps.add(r)
527 return subset & ps
527 return subset & ps
528
528
529 @predicate('author(string)', safe=True)
529 @predicate('author(string)', safe=True)
530 def author(repo, subset, x):
530 def author(repo, subset, x):
531 """Alias for ``user(string)``.
531 """Alias for ``user(string)``.
532 """
532 """
533 # i18n: "author" is a keyword
533 # i18n: "author" is a keyword
534 n = encoding.lower(getstring(x, _("author requires a string")))
534 n = encoding.lower(getstring(x, _("author requires a string")))
535 kind, pattern, matcher = _substringmatcher(n)
535 kind, pattern, matcher = _substringmatcher(n)
536 return subset.filter(lambda x: matcher(encoding.lower(repo[x].user())),
536 return subset.filter(lambda x: matcher(encoding.lower(repo[x].user())),
537 condrepr=('<user %r>', n))
537 condrepr=('<user %r>', n))
538
538
539 @predicate('bisect(string)', safe=True)
539 @predicate('bisect(string)', safe=True)
540 def bisect(repo, subset, x):
540 def bisect(repo, subset, x):
541 """Changesets marked in the specified bisect status:
541 """Changesets marked in the specified bisect status:
542
542
543 - ``good``, ``bad``, ``skip``: csets explicitly marked as good/bad/skip
543 - ``good``, ``bad``, ``skip``: csets explicitly marked as good/bad/skip
544 - ``goods``, ``bads`` : csets topologically good/bad
544 - ``goods``, ``bads`` : csets topologically good/bad
545 - ``range`` : csets taking part in the bisection
545 - ``range`` : csets taking part in the bisection
546 - ``pruned`` : csets that are goods, bads or skipped
546 - ``pruned`` : csets that are goods, bads or skipped
547 - ``untested`` : csets whose fate is yet unknown
547 - ``untested`` : csets whose fate is yet unknown
548 - ``ignored`` : csets ignored due to DAG topology
548 - ``ignored`` : csets ignored due to DAG topology
549 - ``current`` : the cset currently being bisected
549 - ``current`` : the cset currently being bisected
550 """
550 """
551 # i18n: "bisect" is a keyword
551 # i18n: "bisect" is a keyword
552 status = getstring(x, _("bisect requires a string")).lower()
552 status = getstring(x, _("bisect requires a string")).lower()
553 state = set(hbisect.get(repo, status))
553 state = set(hbisect.get(repo, status))
554 return subset & state
554 return subset & state
555
555
556 # Backward-compatibility
556 # Backward-compatibility
557 # - no help entry so that we do not advertise it any more
557 # - no help entry so that we do not advertise it any more
558 @predicate('bisected', safe=True)
558 @predicate('bisected', safe=True)
559 def bisected(repo, subset, x):
559 def bisected(repo, subset, x):
560 return bisect(repo, subset, x)
560 return bisect(repo, subset, x)
561
561
562 @predicate('bookmark([name])', safe=True)
562 @predicate('bookmark([name])', safe=True)
563 def bookmark(repo, subset, x):
563 def bookmark(repo, subset, x):
564 """The named bookmark or all bookmarks.
564 """The named bookmark or all bookmarks.
565
565
566 If `name` starts with `re:`, the remainder of the name is treated as
566 If `name` starts with `re:`, the remainder of the name is treated as
567 a regular expression. To match a bookmark that actually starts with `re:`,
567 a regular expression. To match a bookmark that actually starts with `re:`,
568 use the prefix `literal:`.
568 use the prefix `literal:`.
569 """
569 """
570 # i18n: "bookmark" is a keyword
570 # i18n: "bookmark" is a keyword
571 args = getargs(x, 0, 1, _('bookmark takes one or no arguments'))
571 args = getargs(x, 0, 1, _('bookmark takes one or no arguments'))
572 if args:
572 if args:
573 bm = getstring(args[0],
573 bm = getstring(args[0],
574 # i18n: "bookmark" is a keyword
574 # i18n: "bookmark" is a keyword
575 _('the argument to bookmark must be a string'))
575 _('the argument to bookmark must be a string'))
576 kind, pattern, matcher = util.stringmatcher(bm)
576 kind, pattern, matcher = util.stringmatcher(bm)
577 bms = set()
577 bms = set()
578 if kind == 'literal':
578 if kind == 'literal':
579 bmrev = repo._bookmarks.get(pattern, None)
579 bmrev = repo._bookmarks.get(pattern, None)
580 if not bmrev:
580 if not bmrev:
581 raise error.RepoLookupError(_("bookmark '%s' does not exist")
581 raise error.RepoLookupError(_("bookmark '%s' does not exist")
582 % pattern)
582 % pattern)
583 bms.add(repo[bmrev].rev())
583 bms.add(repo[bmrev].rev())
584 else:
584 else:
585 matchrevs = set()
585 matchrevs = set()
586 for name, bmrev in repo._bookmarks.iteritems():
586 for name, bmrev in repo._bookmarks.iteritems():
587 if matcher(name):
587 if matcher(name):
588 matchrevs.add(bmrev)
588 matchrevs.add(bmrev)
589 if not matchrevs:
589 if not matchrevs:
590 raise error.RepoLookupError(_("no bookmarks exist"
590 raise error.RepoLookupError(_("no bookmarks exist"
591 " that match '%s'") % pattern)
591 " that match '%s'") % pattern)
592 for bmrev in matchrevs:
592 for bmrev in matchrevs:
593 bms.add(repo[bmrev].rev())
593 bms.add(repo[bmrev].rev())
594 else:
594 else:
595 bms = set([repo[r].rev()
595 bms = set([repo[r].rev()
596 for r in repo._bookmarks.values()])
596 for r in repo._bookmarks.values()])
597 bms -= set([node.nullrev])
597 bms -= set([node.nullrev])
598 return subset & bms
598 return subset & bms
599
599
600 @predicate('branch(string or set)', safe=True)
600 @predicate('branch(string or set)', safe=True)
601 def branch(repo, subset, x):
601 def branch(repo, subset, x):
602 """
602 """
603 All changesets belonging to the given branch or the branches of the given
603 All changesets belonging to the given branch or the branches of the given
604 changesets.
604 changesets.
605
605
606 If `string` starts with `re:`, the remainder of the name is treated as
606 If `string` starts with `re:`, the remainder of the name is treated as
607 a regular expression. To match a branch that actually starts with `re:`,
607 a regular expression. To match a branch that actually starts with `re:`,
608 use the prefix `literal:`.
608 use the prefix `literal:`.
609 """
609 """
610 getbi = repo.revbranchcache().branchinfo
610 getbi = repo.revbranchcache().branchinfo
611
611
612 try:
612 try:
613 b = getstring(x, '')
613 b = getstring(x, '')
614 except error.ParseError:
614 except error.ParseError:
615 # not a string, but another revspec, e.g. tip()
615 # not a string, but another revspec, e.g. tip()
616 pass
616 pass
617 else:
617 else:
618 kind, pattern, matcher = util.stringmatcher(b)
618 kind, pattern, matcher = util.stringmatcher(b)
619 if kind == 'literal':
619 if kind == 'literal':
620 # note: falls through to the revspec case if no branch with
620 # note: falls through to the revspec case if no branch with
621 # this name exists and pattern kind is not specified explicitly
621 # this name exists and pattern kind is not specified explicitly
622 if pattern in repo.branchmap():
622 if pattern in repo.branchmap():
623 return subset.filter(lambda r: matcher(getbi(r)[0]),
623 return subset.filter(lambda r: matcher(getbi(r)[0]),
624 condrepr=('<branch %r>', b))
624 condrepr=('<branch %r>', b))
625 if b.startswith('literal:'):
625 if b.startswith('literal:'):
626 raise error.RepoLookupError(_("branch '%s' does not exist")
626 raise error.RepoLookupError(_("branch '%s' does not exist")
627 % pattern)
627 % pattern)
628 else:
628 else:
629 return subset.filter(lambda r: matcher(getbi(r)[0]),
629 return subset.filter(lambda r: matcher(getbi(r)[0]),
630 condrepr=('<branch %r>', b))
630 condrepr=('<branch %r>', b))
631
631
632 s = getset(repo, fullreposet(repo), x)
632 s = getset(repo, fullreposet(repo), x)
633 b = set()
633 b = set()
634 for r in s:
634 for r in s:
635 b.add(getbi(r)[0])
635 b.add(getbi(r)[0])
636 c = s.__contains__
636 c = s.__contains__
637 return subset.filter(lambda r: c(r) or getbi(r)[0] in b,
637 return subset.filter(lambda r: c(r) or getbi(r)[0] in b,
638 condrepr=lambda: '<branch %r>' % sorted(b))
638 condrepr=lambda: '<branch %r>' % sorted(b))
639
639
640 @predicate('bumped()', safe=True)
640 @predicate('bumped()', safe=True)
641 def bumped(repo, subset, x):
641 def bumped(repo, subset, x):
642 """Mutable changesets marked as successors of public changesets.
642 """Mutable changesets marked as successors of public changesets.
643
643
644 Only non-public and non-obsolete changesets can be `bumped`.
644 Only non-public and non-obsolete changesets can be `bumped`.
645 """
645 """
646 # i18n: "bumped" is a keyword
646 # i18n: "bumped" is a keyword
647 getargs(x, 0, 0, _("bumped takes no arguments"))
647 getargs(x, 0, 0, _("bumped takes no arguments"))
648 bumped = obsmod.getrevs(repo, 'bumped')
648 bumped = obsmod.getrevs(repo, 'bumped')
649 return subset & bumped
649 return subset & bumped
650
650
651 @predicate('bundle()', safe=True)
651 @predicate('bundle()', safe=True)
652 def bundle(repo, subset, x):
652 def bundle(repo, subset, x):
653 """Changesets in the bundle.
653 """Changesets in the bundle.
654
654
655 Bundle must be specified by the -R option."""
655 Bundle must be specified by the -R option."""
656
656
657 try:
657 try:
658 bundlerevs = repo.changelog.bundlerevs
658 bundlerevs = repo.changelog.bundlerevs
659 except AttributeError:
659 except AttributeError:
660 raise error.Abort(_("no bundle provided - specify with -R"))
660 raise error.Abort(_("no bundle provided - specify with -R"))
661 return subset & bundlerevs
661 return subset & bundlerevs
662
662
663 def checkstatus(repo, subset, pat, field):
663 def checkstatus(repo, subset, pat, field):
664 hasset = matchmod.patkind(pat) == 'set'
664 hasset = matchmod.patkind(pat) == 'set'
665
665
666 mcache = [None]
666 mcache = [None]
667 def matches(x):
667 def matches(x):
668 c = repo[x]
668 c = repo[x]
669 if not mcache[0] or hasset:
669 if not mcache[0] or hasset:
670 mcache[0] = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=c)
670 mcache[0] = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=c)
671 m = mcache[0]
671 m = mcache[0]
672 fname = None
672 fname = None
673 if not m.anypats() and len(m.files()) == 1:
673 if not m.anypats() and len(m.files()) == 1:
674 fname = m.files()[0]
674 fname = m.files()[0]
675 if fname is not None:
675 if fname is not None:
676 if fname not in c.files():
676 if fname not in c.files():
677 return False
677 return False
678 else:
678 else:
679 for f in c.files():
679 for f in c.files():
680 if m(f):
680 if m(f):
681 break
681 break
682 else:
682 else:
683 return False
683 return False
684 files = repo.status(c.p1().node(), c.node())[field]
684 files = repo.status(c.p1().node(), c.node())[field]
685 if fname is not None:
685 if fname is not None:
686 if fname in files:
686 if fname in files:
687 return True
687 return True
688 else:
688 else:
689 for f in files:
689 for f in files:
690 if m(f):
690 if m(f):
691 return True
691 return True
692
692
693 return subset.filter(matches, condrepr=('<status[%r] %r>', field, pat))
693 return subset.filter(matches, condrepr=('<status[%r] %r>', field, pat))
694
694
695 def _children(repo, narrow, parentset):
695 def _children(repo, narrow, parentset):
696 if not parentset:
696 if not parentset:
697 return baseset()
697 return baseset()
698 cs = set()
698 cs = set()
699 pr = repo.changelog.parentrevs
699 pr = repo.changelog.parentrevs
700 minrev = parentset.min()
700 minrev = parentset.min()
701 for r in narrow:
701 for r in narrow:
702 if r <= minrev:
702 if r <= minrev:
703 continue
703 continue
704 for p in pr(r):
704 for p in pr(r):
705 if p in parentset:
705 if p in parentset:
706 cs.add(r)
706 cs.add(r)
707 # XXX using a set to feed the baseset is wrong. Sets are not ordered.
707 # XXX using a set to feed the baseset is wrong. Sets are not ordered.
708 # This does not break because of other fullreposet misbehavior.
708 # This does not break because of other fullreposet misbehavior.
709 return baseset(cs)
709 return baseset(cs)
710
710
711 @predicate('children(set)', safe=True)
711 @predicate('children(set)', safe=True)
712 def children(repo, subset, x):
712 def children(repo, subset, x):
713 """Child changesets of changesets in set.
713 """Child changesets of changesets in set.
714 """
714 """
715 s = getset(repo, fullreposet(repo), x)
715 s = getset(repo, fullreposet(repo), x)
716 cs = _children(repo, subset, s)
716 cs = _children(repo, subset, s)
717 return subset & cs
717 return subset & cs
718
718
719 @predicate('closed()', safe=True)
719 @predicate('closed()', safe=True)
720 def closed(repo, subset, x):
720 def closed(repo, subset, x):
721 """Changeset is closed.
721 """Changeset is closed.
722 """
722 """
723 # i18n: "closed" is a keyword
723 # i18n: "closed" is a keyword
724 getargs(x, 0, 0, _("closed takes no arguments"))
724 getargs(x, 0, 0, _("closed takes no arguments"))
725 return subset.filter(lambda r: repo[r].closesbranch(),
725 return subset.filter(lambda r: repo[r].closesbranch(),
726 condrepr='<branch closed>')
726 condrepr='<branch closed>')
727
727
728 @predicate('contains(pattern)')
728 @predicate('contains(pattern)')
729 def contains(repo, subset, x):
729 def contains(repo, subset, x):
730 """The revision's manifest contains a file matching pattern (but might not
730 """The revision's manifest contains a file matching pattern (but might not
731 modify it). See :hg:`help patterns` for information about file patterns.
731 modify it). See :hg:`help patterns` for information about file patterns.
732
732
733 The pattern without explicit kind like ``glob:`` is expected to be
733 The pattern without explicit kind like ``glob:`` is expected to be
734 relative to the current directory and match against a file exactly
734 relative to the current directory and match against a file exactly
735 for efficiency.
735 for efficiency.
736 """
736 """
737 # i18n: "contains" is a keyword
737 # i18n: "contains" is a keyword
738 pat = getstring(x, _("contains requires a pattern"))
738 pat = getstring(x, _("contains requires a pattern"))
739
739
740 def matches(x):
740 def matches(x):
741 if not matchmod.patkind(pat):
741 if not matchmod.patkind(pat):
742 pats = pathutil.canonpath(repo.root, repo.getcwd(), pat)
742 pats = pathutil.canonpath(repo.root, repo.getcwd(), pat)
743 if pats in repo[x]:
743 if pats in repo[x]:
744 return True
744 return True
745 else:
745 else:
746 c = repo[x]
746 c = repo[x]
747 m = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=c)
747 m = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=c)
748 for f in c.manifest():
748 for f in c.manifest():
749 if m(f):
749 if m(f):
750 return True
750 return True
751 return False
751 return False
752
752
753 return subset.filter(matches, condrepr=('<contains %r>', pat))
753 return subset.filter(matches, condrepr=('<contains %r>', pat))
754
754
755 @predicate('converted([id])', safe=True)
755 @predicate('converted([id])', safe=True)
756 def converted(repo, subset, x):
756 def converted(repo, subset, x):
757 """Changesets converted from the given identifier in the old repository if
757 """Changesets converted from the given identifier in the old repository if
758 present, or all converted changesets if no identifier is specified.
758 present, or all converted changesets if no identifier is specified.
759 """
759 """
760
760
761 # There is exactly no chance of resolving the revision, so do a simple
761 # There is exactly no chance of resolving the revision, so do a simple
762 # string compare and hope for the best
762 # string compare and hope for the best
763
763
764 rev = None
764 rev = None
765 # i18n: "converted" is a keyword
765 # i18n: "converted" is a keyword
766 l = getargs(x, 0, 1, _('converted takes one or no arguments'))
766 l = getargs(x, 0, 1, _('converted takes one or no arguments'))
767 if l:
767 if l:
768 # i18n: "converted" is a keyword
768 # i18n: "converted" is a keyword
769 rev = getstring(l[0], _('converted requires a revision'))
769 rev = getstring(l[0], _('converted requires a revision'))
770
770
771 def _matchvalue(r):
771 def _matchvalue(r):
772 source = repo[r].extra().get('convert_revision', None)
772 source = repo[r].extra().get('convert_revision', None)
773 return source is not None and (rev is None or source.startswith(rev))
773 return source is not None and (rev is None or source.startswith(rev))
774
774
775 return subset.filter(lambda r: _matchvalue(r),
775 return subset.filter(lambda r: _matchvalue(r),
776 condrepr=('<converted %r>', rev))
776 condrepr=('<converted %r>', rev))
777
777
778 @predicate('date(interval)', safe=True)
778 @predicate('date(interval)', safe=True)
779 def date(repo, subset, x):
779 def date(repo, subset, x):
780 """Changesets within the interval, see :hg:`help dates`.
780 """Changesets within the interval, see :hg:`help dates`.
781 """
781 """
782 # i18n: "date" is a keyword
782 # i18n: "date" is a keyword
783 ds = getstring(x, _("date requires a string"))
783 ds = getstring(x, _("date requires a string"))
784 dm = util.matchdate(ds)
784 dm = util.matchdate(ds)
785 return subset.filter(lambda x: dm(repo[x].date()[0]),
785 return subset.filter(lambda x: dm(repo[x].date()[0]),
786 condrepr=('<date %r>', ds))
786 condrepr=('<date %r>', ds))
787
787
788 @predicate('desc(string)', safe=True)
788 @predicate('desc(string)', safe=True)
789 def desc(repo, subset, x):
789 def desc(repo, subset, x):
790 """Search commit message for string. The match is case-insensitive.
790 """Search commit message for string. The match is case-insensitive.
791 """
791 """
792 # i18n: "desc" is a keyword
792 # i18n: "desc" is a keyword
793 ds = encoding.lower(getstring(x, _("desc requires a string")))
793 ds = encoding.lower(getstring(x, _("desc requires a string")))
794
794
795 def matches(x):
795 def matches(x):
796 c = repo[x]
796 c = repo[x]
797 return ds in encoding.lower(c.description())
797 return ds in encoding.lower(c.description())
798
798
799 return subset.filter(matches, condrepr=('<desc %r>', ds))
799 return subset.filter(matches, condrepr=('<desc %r>', ds))
800
800
801 def _descendants(repo, subset, x, followfirst=False):
801 def _descendants(repo, subset, x, followfirst=False):
802 roots = getset(repo, fullreposet(repo), x)
802 roots = getset(repo, fullreposet(repo), x)
803 if not roots:
803 if not roots:
804 return baseset()
804 return baseset()
805 s = _revdescendants(repo, roots, followfirst)
805 s = _revdescendants(repo, roots, followfirst)
806
806
807 # Both sets need to be ascending in order to lazily return the union
807 # Both sets need to be ascending in order to lazily return the union
808 # in the correct order.
808 # in the correct order.
809 base = subset & roots
809 base = subset & roots
810 desc = subset & s
810 desc = subset & s
811 result = base + desc
811 result = base + desc
812 if subset.isascending():
812 if subset.isascending():
813 result.sort()
813 result.sort()
814 elif subset.isdescending():
814 elif subset.isdescending():
815 result.sort(reverse=True)
815 result.sort(reverse=True)
816 else:
816 else:
817 result = subset & result
817 result = subset & result
818 return result
818 return result
819
819
820 @predicate('descendants(set)', safe=True)
820 @predicate('descendants(set)', safe=True)
821 def descendants(repo, subset, x):
821 def descendants(repo, subset, x):
822 """Changesets which are descendants of changesets in set.
822 """Changesets which are descendants of changesets in set.
823 """
823 """
824 return _descendants(repo, subset, x)
824 return _descendants(repo, subset, x)
825
825
826 @predicate('_firstdescendants', safe=True)
826 @predicate('_firstdescendants', safe=True)
827 def _firstdescendants(repo, subset, x):
827 def _firstdescendants(repo, subset, x):
828 # ``_firstdescendants(set)``
828 # ``_firstdescendants(set)``
829 # Like ``descendants(set)`` but follows only the first parents.
829 # Like ``descendants(set)`` but follows only the first parents.
830 return _descendants(repo, subset, x, followfirst=True)
830 return _descendants(repo, subset, x, followfirst=True)
831
831
832 @predicate('destination([set])', safe=True)
832 @predicate('destination([set])', safe=True)
833 def destination(repo, subset, x):
833 def destination(repo, subset, x):
834 """Changesets that were created by a graft, transplant or rebase operation,
834 """Changesets that were created by a graft, transplant or rebase operation,
835 with the given revisions specified as the source. Omitting the optional set
835 with the given revisions specified as the source. Omitting the optional set
836 is the same as passing all().
836 is the same as passing all().
837 """
837 """
838 if x is not None:
838 if x is not None:
839 sources = getset(repo, fullreposet(repo), x)
839 sources = getset(repo, fullreposet(repo), x)
840 else:
840 else:
841 sources = fullreposet(repo)
841 sources = fullreposet(repo)
842
842
843 dests = set()
843 dests = set()
844
844
845 # subset contains all of the possible destinations that can be returned, so
845 # subset contains all of the possible destinations that can be returned, so
846 # iterate over them and see if their source(s) were provided in the arg set.
846 # iterate over them and see if their source(s) were provided in the arg set.
847 # Even if the immediate src of r is not in the arg set, src's source (or
847 # Even if the immediate src of r is not in the arg set, src's source (or
848 # further back) may be. Scanning back further than the immediate src allows
848 # further back) may be. Scanning back further than the immediate src allows
849 # transitive transplants and rebases to yield the same results as transitive
849 # transitive transplants and rebases to yield the same results as transitive
850 # grafts.
850 # grafts.
851 for r in subset:
851 for r in subset:
852 src = _getrevsource(repo, r)
852 src = _getrevsource(repo, r)
853 lineage = None
853 lineage = None
854
854
855 while src is not None:
855 while src is not None:
856 if lineage is None:
856 if lineage is None:
857 lineage = list()
857 lineage = list()
858
858
859 lineage.append(r)
859 lineage.append(r)
860
860
861 # The visited lineage is a match if the current source is in the arg
861 # The visited lineage is a match if the current source is in the arg
862 # set. Since every candidate dest is visited by way of iterating
862 # set. Since every candidate dest is visited by way of iterating
863 # subset, any dests further back in the lineage will be tested by a
863 # subset, any dests further back in the lineage will be tested by a
864 # different iteration over subset. Likewise, if the src was already
864 # different iteration over subset. Likewise, if the src was already
865 # selected, the current lineage can be selected without going back
865 # selected, the current lineage can be selected without going back
866 # further.
866 # further.
867 if src in sources or src in dests:
867 if src in sources or src in dests:
868 dests.update(lineage)
868 dests.update(lineage)
869 break
869 break
870
870
871 r = src
871 r = src
872 src = _getrevsource(repo, r)
872 src = _getrevsource(repo, r)
873
873
874 return subset.filter(dests.__contains__,
874 return subset.filter(dests.__contains__,
875 condrepr=lambda: '<destination %r>' % sorted(dests))
875 condrepr=lambda: '<destination %r>' % sorted(dests))
876
876
877 @predicate('divergent()', safe=True)
877 @predicate('divergent()', safe=True)
878 def divergent(repo, subset, x):
878 def divergent(repo, subset, x):
879 """
879 """
880 Final successors of changesets with an alternative set of final successors.
880 Final successors of changesets with an alternative set of final successors.
881 """
881 """
882 # i18n: "divergent" is a keyword
882 # i18n: "divergent" is a keyword
883 getargs(x, 0, 0, _("divergent takes no arguments"))
883 getargs(x, 0, 0, _("divergent takes no arguments"))
884 divergent = obsmod.getrevs(repo, 'divergent')
884 divergent = obsmod.getrevs(repo, 'divergent')
885 return subset & divergent
885 return subset & divergent
886
886
887 @predicate('extinct()', safe=True)
887 @predicate('extinct()', safe=True)
888 def extinct(repo, subset, x):
888 def extinct(repo, subset, x):
889 """Obsolete changesets with obsolete descendants only.
889 """Obsolete changesets with obsolete descendants only.
890 """
890 """
891 # i18n: "extinct" is a keyword
891 # i18n: "extinct" is a keyword
892 getargs(x, 0, 0, _("extinct takes no arguments"))
892 getargs(x, 0, 0, _("extinct takes no arguments"))
893 extincts = obsmod.getrevs(repo, 'extinct')
893 extincts = obsmod.getrevs(repo, 'extinct')
894 return subset & extincts
894 return subset & extincts
895
895
896 @predicate('extra(label, [value])', safe=True)
896 @predicate('extra(label, [value])', safe=True)
897 def extra(repo, subset, x):
897 def extra(repo, subset, x):
898 """Changesets with the given label in the extra metadata, with the given
898 """Changesets with the given label in the extra metadata, with the given
899 optional value.
899 optional value.
900
900
901 If `value` starts with `re:`, the remainder of the value is treated as
901 If `value` starts with `re:`, the remainder of the value is treated as
902 a regular expression. To match a value that actually starts with `re:`,
902 a regular expression. To match a value that actually starts with `re:`,
903 use the prefix `literal:`.
903 use the prefix `literal:`.
904 """
904 """
905 args = getargsdict(x, 'extra', 'label value')
905 args = getargsdict(x, 'extra', 'label value')
906 if 'label' not in args:
906 if 'label' not in args:
907 # i18n: "extra" is a keyword
907 # i18n: "extra" is a keyword
908 raise error.ParseError(_('extra takes at least 1 argument'))
908 raise error.ParseError(_('extra takes at least 1 argument'))
909 # i18n: "extra" is a keyword
909 # i18n: "extra" is a keyword
910 label = getstring(args['label'], _('first argument to extra must be '
910 label = getstring(args['label'], _('first argument to extra must be '
911 'a string'))
911 'a string'))
912 value = None
912 value = None
913
913
914 if 'value' in args:
914 if 'value' in args:
915 # i18n: "extra" is a keyword
915 # i18n: "extra" is a keyword
916 value = getstring(args['value'], _('second argument to extra must be '
916 value = getstring(args['value'], _('second argument to extra must be '
917 'a string'))
917 'a string'))
918 kind, value, matcher = util.stringmatcher(value)
918 kind, value, matcher = util.stringmatcher(value)
919
919
920 def _matchvalue(r):
920 def _matchvalue(r):
921 extra = repo[r].extra()
921 extra = repo[r].extra()
922 return label in extra and (value is None or matcher(extra[label]))
922 return label in extra and (value is None or matcher(extra[label]))
923
923
924 return subset.filter(lambda r: _matchvalue(r),
924 return subset.filter(lambda r: _matchvalue(r),
925 condrepr=('<extra[%r] %r>', label, value))
925 condrepr=('<extra[%r] %r>', label, value))
926
926
927 @predicate('filelog(pattern)', safe=True)
927 @predicate('filelog(pattern)', safe=True)
928 def filelog(repo, subset, x):
928 def filelog(repo, subset, x):
929 """Changesets connected to the specified filelog.
929 """Changesets connected to the specified filelog.
930
930
931 For performance reasons, visits only revisions mentioned in the file-level
931 For performance reasons, visits only revisions mentioned in the file-level
932 filelog, rather than filtering through all changesets (much faster, but
932 filelog, rather than filtering through all changesets (much faster, but
933 doesn't include deletes or duplicate changes). For a slower, more accurate
933 doesn't include deletes or duplicate changes). For a slower, more accurate
934 result, use ``file()``.
934 result, use ``file()``.
935
935
936 The pattern without explicit kind like ``glob:`` is expected to be
936 The pattern without explicit kind like ``glob:`` is expected to be
937 relative to the current directory and match against a file exactly
937 relative to the current directory and match against a file exactly
938 for efficiency.
938 for efficiency.
939
939
940 If some linkrev points to revisions filtered by the current repoview, we'll
940 If some linkrev points to revisions filtered by the current repoview, we'll
941 work around it to return a non-filtered value.
941 work around it to return a non-filtered value.
942 """
942 """
943
943
944 # i18n: "filelog" is a keyword
944 # i18n: "filelog" is a keyword
945 pat = getstring(x, _("filelog requires a pattern"))
945 pat = getstring(x, _("filelog requires a pattern"))
946 s = set()
946 s = set()
947 cl = repo.changelog
947 cl = repo.changelog
948
948
949 if not matchmod.patkind(pat):
949 if not matchmod.patkind(pat):
950 f = pathutil.canonpath(repo.root, repo.getcwd(), pat)
950 f = pathutil.canonpath(repo.root, repo.getcwd(), pat)
951 files = [f]
951 files = [f]
952 else:
952 else:
953 m = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=repo[None])
953 m = matchmod.match(repo.root, repo.getcwd(), [pat], ctx=repo[None])
954 files = (f for f in repo[None] if m(f))
954 files = (f for f in repo[None] if m(f))
955
955
956 for f in files:
956 for f in files:
957 fl = repo.file(f)
957 fl = repo.file(f)
958 known = {}
958 known = {}
959 scanpos = 0
959 scanpos = 0
960 for fr in list(fl):
960 for fr in list(fl):
961 fn = fl.node(fr)
961 fn = fl.node(fr)
962 if fn in known:
962 if fn in known:
963 s.add(known[fn])
963 s.add(known[fn])
964 continue
964 continue
965
965
966 lr = fl.linkrev(fr)
966 lr = fl.linkrev(fr)
967 if lr in cl:
967 if lr in cl:
968 s.add(lr)
968 s.add(lr)
969 elif scanpos is not None:
969 elif scanpos is not None:
970 # lowest matching changeset is filtered, scan further
970 # lowest matching changeset is filtered, scan further
971 # ahead in changelog
971 # ahead in changelog
972 start = max(lr, scanpos) + 1
972 start = max(lr, scanpos) + 1
973 scanpos = None
973 scanpos = None
974 for r in cl.revs(start):
974 for r in cl.revs(start):
975 # minimize parsing of non-matching entries
975 # minimize parsing of non-matching entries
976 if f in cl.revision(r) and f in cl.readfiles(r):
976 if f in cl.revision(r) and f in cl.readfiles(r):
977 try:
977 try:
978 # try to use manifest delta fastpath
978 # try to use manifest delta fastpath
979 n = repo[r].filenode(f)
979 n = repo[r].filenode(f)
980 if n not in known:
980 if n not in known:
981 if n == fn:
981 if n == fn:
982 s.add(r)
982 s.add(r)
983 scanpos = r
983 scanpos = r
984 break
984 break
985 else:
985 else:
986 known[n] = r
986 known[n] = r
987 except error.ManifestLookupError:
987 except error.ManifestLookupError:
988 # deletion in changelog
988 # deletion in changelog
989 continue
989 continue
990
990
991 return subset & s
991 return subset & s
992
992
993 @predicate('first(set, [n])', safe=True)
993 @predicate('first(set, [n])', safe=True)
994 def first(repo, subset, x):
994 def first(repo, subset, x):
995 """An alias for limit().
995 """An alias for limit().
996 """
996 """
997 return limit(repo, subset, x)
997 return limit(repo, subset, x)
998
998
999 def _follow(repo, subset, x, name, followfirst=False):
999 def _follow(repo, subset, x, name, followfirst=False):
1000 l = getargs(x, 0, 1, _("%s takes no arguments or a pattern") % name)
1000 l = getargs(x, 0, 1, _("%s takes no arguments or a pattern") % name)
1001 c = repo['.']
1001 c = repo['.']
1002 if l:
1002 if l:
1003 x = getstring(l[0], _("%s expected a pattern") % name)
1003 x = getstring(l[0], _("%s expected a pattern") % name)
1004 matcher = matchmod.match(repo.root, repo.getcwd(), [x],
1004 matcher = matchmod.match(repo.root, repo.getcwd(), [x],
1005 ctx=repo[None], default='path')
1005 ctx=repo[None], default='path')
1006
1006
1007 files = c.manifest().walk(matcher)
1007 files = c.manifest().walk(matcher)
1008
1008
1009 s = set()
1009 s = set()
1010 for fname in files:
1010 for fname in files:
1011 fctx = c[fname]
1011 fctx = c[fname]
1012 s = s.union(set(c.rev() for c in fctx.ancestors(followfirst)))
1012 s = s.union(set(c.rev() for c in fctx.ancestors(followfirst)))
1013 # include the revision responsible for the most recent version
1013 # include the revision responsible for the most recent version
1014 s.add(fctx.introrev())
1014 s.add(fctx.introrev())
1015 else:
1015 else:
1016 s = _revancestors(repo, baseset([c.rev()]), followfirst)
1016 s = _revancestors(repo, baseset([c.rev()]), followfirst)
1017
1017
1018 return subset & s
1018 return subset & s
1019
1019
1020 @predicate('follow([pattern])', safe=True)
1020 @predicate('follow([pattern])', safe=True)
1021 def follow(repo, subset, x):
1021 def follow(repo, subset, x):
1022 """
1022 """
1023 An alias for ``::.`` (ancestors of the working directory's first parent).
1023 An alias for ``::.`` (ancestors of the working directory's first parent).
1024 If pattern is specified, the histories of files matching given
1024 If pattern is specified, the histories of files matching given
1025 pattern is followed, including copies.
1025 pattern is followed, including copies.
1026 """
1026 """
1027 return _follow(repo, subset, x, 'follow')
1027 return _follow(repo, subset, x, 'follow')
1028
1028
1029 @predicate('_followfirst', safe=True)
1029 @predicate('_followfirst', safe=True)
1030 def _followfirst(repo, subset, x):
1030 def _followfirst(repo, subset, x):
1031 # ``followfirst([pattern])``
1031 # ``followfirst([pattern])``
1032 # Like ``follow([pattern])`` but follows only the first parent of
1032 # Like ``follow([pattern])`` but follows only the first parent of
1033 # every revisions or files revisions.
1033 # every revisions or files revisions.
1034 return _follow(repo, subset, x, '_followfirst', followfirst=True)
1034 return _follow(repo, subset, x, '_followfirst', followfirst=True)
1035
1035
1036 @predicate('all()', safe=True)
1036 @predicate('all()', safe=True)
1037 def getall(repo, subset, x):
1037 def getall(repo, subset, x):
1038 """All changesets, the same as ``0:tip``.
1038 """All changesets, the same as ``0:tip``.
1039 """
1039 """
1040 # i18n: "all" is a keyword
1040 # i18n: "all" is a keyword
1041 getargs(x, 0, 0, _("all takes no arguments"))
1041 getargs(x, 0, 0, _("all takes no arguments"))
1042 return subset & spanset(repo) # drop "null" if any
1042 return subset & spanset(repo) # drop "null" if any
1043
1043
1044 @predicate('grep(regex)')
1044 @predicate('grep(regex)')
1045 def grep(repo, subset, x):
1045 def grep(repo, subset, x):
1046 """Like ``keyword(string)`` but accepts a regex. Use ``grep(r'...')``
1046 """Like ``keyword(string)`` but accepts a regex. Use ``grep(r'...')``
1047 to ensure special escape characters are handled correctly. Unlike
1047 to ensure special escape characters are handled correctly. Unlike
1048 ``keyword(string)``, the match is case-sensitive.
1048 ``keyword(string)``, the match is case-sensitive.
1049 """
1049 """
1050 try:
1050 try:
1051 # i18n: "grep" is a keyword
1051 # i18n: "grep" is a keyword
1052 gr = re.compile(getstring(x, _("grep requires a string")))
1052 gr = re.compile(getstring(x, _("grep requires a string")))
1053 except re.error as e:
1053 except re.error as e:
1054 raise error.ParseError(_('invalid match pattern: %s') % e)
1054 raise error.ParseError(_('invalid match pattern: %s') % e)
1055
1055
1056 def matches(x):
1056 def matches(x):
1057 c = repo[x]
1057 c = repo[x]
1058 for e in c.files() + [c.user(), c.description()]:
1058 for e in c.files() + [c.user(), c.description()]:
1059 if gr.search(e):
1059 if gr.search(e):
1060 return True
1060 return True
1061 return False
1061 return False
1062
1062
1063 return subset.filter(matches, condrepr=('<grep %r>', gr.pattern))
1063 return subset.filter(matches, condrepr=('<grep %r>', gr.pattern))
1064
1064
1065 @predicate('_matchfiles', safe=True)
1065 @predicate('_matchfiles', safe=True)
1066 def _matchfiles(repo, subset, x):
1066 def _matchfiles(repo, subset, x):
1067 # _matchfiles takes a revset list of prefixed arguments:
1067 # _matchfiles takes a revset list of prefixed arguments:
1068 #
1068 #
1069 # [p:foo, i:bar, x:baz]
1069 # [p:foo, i:bar, x:baz]
1070 #
1070 #
1071 # builds a match object from them and filters subset. Allowed
1071 # builds a match object from them and filters subset. Allowed
1072 # prefixes are 'p:' for regular patterns, 'i:' for include
1072 # prefixes are 'p:' for regular patterns, 'i:' for include
1073 # patterns and 'x:' for exclude patterns. Use 'r:' prefix to pass
1073 # patterns and 'x:' for exclude patterns. Use 'r:' prefix to pass
1074 # a revision identifier, or the empty string to reference the
1074 # a revision identifier, or the empty string to reference the
1075 # working directory, from which the match object is
1075 # working directory, from which the match object is
1076 # initialized. Use 'd:' to set the default matching mode, default
1076 # initialized. Use 'd:' to set the default matching mode, default
1077 # to 'glob'. At most one 'r:' and 'd:' argument can be passed.
1077 # to 'glob'. At most one 'r:' and 'd:' argument can be passed.
1078
1078
1079 l = getargs(x, 1, -1, "_matchfiles requires at least one argument")
1079 l = getargs(x, 1, -1, "_matchfiles requires at least one argument")
1080 pats, inc, exc = [], [], []
1080 pats, inc, exc = [], [], []
1081 rev, default = None, None
1081 rev, default = None, None
1082 for arg in l:
1082 for arg in l:
1083 s = getstring(arg, "_matchfiles requires string arguments")
1083 s = getstring(arg, "_matchfiles requires string arguments")
1084 prefix, value = s[:2], s[2:]
1084 prefix, value = s[:2], s[2:]
1085 if prefix == 'p:':
1085 if prefix == 'p:':
1086 pats.append(value)
1086 pats.append(value)
1087 elif prefix == 'i:':
1087 elif prefix == 'i:':
1088 inc.append(value)
1088 inc.append(value)
1089 elif prefix == 'x:':
1089 elif prefix == 'x:':
1090 exc.append(value)
1090 exc.append(value)
1091 elif prefix == 'r:':
1091 elif prefix == 'r:':
1092 if rev is not None:
1092 if rev is not None:
1093 raise error.ParseError('_matchfiles expected at most one '
1093 raise error.ParseError('_matchfiles expected at most one '
1094 'revision')
1094 'revision')
1095 if value != '': # empty means working directory; leave rev as None
1095 if value != '': # empty means working directory; leave rev as None
1096 rev = value
1096 rev = value
1097 elif prefix == 'd:':
1097 elif prefix == 'd:':
1098 if default is not None:
1098 if default is not None:
1099 raise error.ParseError('_matchfiles expected at most one '
1099 raise error.ParseError('_matchfiles expected at most one '
1100 'default mode')
1100 'default mode')
1101 default = value
1101 default = value
1102 else:
1102 else:
1103 raise error.ParseError('invalid _matchfiles prefix: %s' % prefix)
1103 raise error.ParseError('invalid _matchfiles prefix: %s' % prefix)
1104 if not default:
1104 if not default:
1105 default = 'glob'
1105 default = 'glob'
1106
1106
1107 m = matchmod.match(repo.root, repo.getcwd(), pats, include=inc,
1107 m = matchmod.match(repo.root, repo.getcwd(), pats, include=inc,
1108 exclude=exc, ctx=repo[rev], default=default)
1108 exclude=exc, ctx=repo[rev], default=default)
1109
1109
1110 # This directly read the changelog data as creating changectx for all
1110 # This directly read the changelog data as creating changectx for all
1111 # revisions is quite expensive.
1111 # revisions is quite expensive.
1112 getfiles = repo.changelog.readfiles
1112 getfiles = repo.changelog.readfiles
1113 wdirrev = node.wdirrev
1113 wdirrev = node.wdirrev
1114 def matches(x):
1114 def matches(x):
1115 if x == wdirrev:
1115 if x == wdirrev:
1116 files = repo[x].files()
1116 files = repo[x].files()
1117 else:
1117 else:
1118 files = getfiles(x)
1118 files = getfiles(x)
1119 for f in files:
1119 for f in files:
1120 if m(f):
1120 if m(f):
1121 return True
1121 return True
1122 return False
1122 return False
1123
1123
1124 return subset.filter(matches,
1124 return subset.filter(matches,
1125 condrepr=('<matchfiles patterns=%r, include=%r '
1125 condrepr=('<matchfiles patterns=%r, include=%r '
1126 'exclude=%r, default=%r, rev=%r>',
1126 'exclude=%r, default=%r, rev=%r>',
1127 pats, inc, exc, default, rev))
1127 pats, inc, exc, default, rev))
1128
1128
1129 @predicate('file(pattern)', safe=True)
1129 @predicate('file(pattern)', safe=True)
1130 def hasfile(repo, subset, x):
1130 def hasfile(repo, subset, x):
1131 """Changesets affecting files matched by pattern.
1131 """Changesets affecting files matched by pattern.
1132
1132
1133 For a faster but less accurate result, consider using ``filelog()``
1133 For a faster but less accurate result, consider using ``filelog()``
1134 instead.
1134 instead.
1135
1135
1136 This predicate uses ``glob:`` as the default kind of pattern.
1136 This predicate uses ``glob:`` as the default kind of pattern.
1137 """
1137 """
1138 # i18n: "file" is a keyword
1138 # i18n: "file" is a keyword
1139 pat = getstring(x, _("file requires a pattern"))
1139 pat = getstring(x, _("file requires a pattern"))
1140 return _matchfiles(repo, subset, ('string', 'p:' + pat))
1140 return _matchfiles(repo, subset, ('string', 'p:' + pat))
1141
1141
1142 @predicate('head()', safe=True)
1142 @predicate('head()', safe=True)
1143 def head(repo, subset, x):
1143 def head(repo, subset, x):
1144 """Changeset is a named branch head.
1144 """Changeset is a named branch head.
1145 """
1145 """
1146 # i18n: "head" is a keyword
1146 # i18n: "head" is a keyword
1147 getargs(x, 0, 0, _("head takes no arguments"))
1147 getargs(x, 0, 0, _("head takes no arguments"))
1148 hs = set()
1148 hs = set()
1149 cl = repo.changelog
1149 cl = repo.changelog
1150 for b, ls in repo.branchmap().iteritems():
1150 for b, ls in repo.branchmap().iteritems():
1151 hs.update(cl.rev(h) for h in ls)
1151 hs.update(cl.rev(h) for h in ls)
1152 # XXX using a set to feed the baseset is wrong. Sets are not ordered.
1152 # XXX using a set to feed the baseset is wrong. Sets are not ordered.
1153 # This does not break because of other fullreposet misbehavior.
1153 # This does not break because of other fullreposet misbehavior.
1154 # XXX We should combine with subset first: 'subset & baseset(...)'. This is
1154 # XXX We should combine with subset first: 'subset & baseset(...)'. This is
1155 # necessary to ensure we preserve the order in subset.
1155 # necessary to ensure we preserve the order in subset.
1156 return baseset(hs) & subset
1156 return baseset(hs) & subset
1157
1157
1158 @predicate('heads(set)', safe=True)
1158 @predicate('heads(set)', safe=True)
1159 def heads(repo, subset, x):
1159 def heads(repo, subset, x):
1160 """Members of set with no children in set.
1160 """Members of set with no children in set.
1161 """
1161 """
1162 s = getset(repo, subset, x)
1162 s = getset(repo, subset, x)
1163 ps = parents(repo, subset, x)
1163 ps = parents(repo, subset, x)
1164 return s - ps
1164 return s - ps
1165
1165
1166 @predicate('hidden()', safe=True)
1166 @predicate('hidden()', safe=True)
1167 def hidden(repo, subset, x):
1167 def hidden(repo, subset, x):
1168 """Hidden changesets.
1168 """Hidden changesets.
1169 """
1169 """
1170 # i18n: "hidden" is a keyword
1170 # i18n: "hidden" is a keyword
1171 getargs(x, 0, 0, _("hidden takes no arguments"))
1171 getargs(x, 0, 0, _("hidden takes no arguments"))
1172 hiddenrevs = repoview.filterrevs(repo, 'visible')
1172 hiddenrevs = repoview.filterrevs(repo, 'visible')
1173 return subset & hiddenrevs
1173 return subset & hiddenrevs
1174
1174
1175 @predicate('keyword(string)', safe=True)
1175 @predicate('keyword(string)', safe=True)
1176 def keyword(repo, subset, x):
1176 def keyword(repo, subset, x):
1177 """Search commit message, user name, and names of changed files for
1177 """Search commit message, user name, and names of changed files for
1178 string. The match is case-insensitive.
1178 string. The match is case-insensitive.
1179 """
1179 """
1180 # i18n: "keyword" is a keyword
1180 # i18n: "keyword" is a keyword
1181 kw = encoding.lower(getstring(x, _("keyword requires a string")))
1181 kw = encoding.lower(getstring(x, _("keyword requires a string")))
1182
1182
1183 def matches(r):
1183 def matches(r):
1184 c = repo[r]
1184 c = repo[r]
1185 return any(kw in encoding.lower(t)
1185 return any(kw in encoding.lower(t)
1186 for t in c.files() + [c.user(), c.description()])
1186 for t in c.files() + [c.user(), c.description()])
1187
1187
1188 return subset.filter(matches, condrepr=('<keyword %r>', kw))
1188 return subset.filter(matches, condrepr=('<keyword %r>', kw))
1189
1189
1190 @predicate('limit(set[, n[, offset]])', safe=True)
1190 @predicate('limit(set[, n[, offset]])', safe=True)
1191 def limit(repo, subset, x):
1191 def limit(repo, subset, x):
1192 """First n members of set, defaulting to 1, starting from offset.
1192 """First n members of set, defaulting to 1, starting from offset.
1193 """
1193 """
1194 args = getargsdict(x, 'limit', 'set n offset')
1194 args = getargsdict(x, 'limit', 'set n offset')
1195 if 'set' not in args:
1195 if 'set' not in args:
1196 # i18n: "limit" is a keyword
1196 # i18n: "limit" is a keyword
1197 raise error.ParseError(_("limit requires one to three arguments"))
1197 raise error.ParseError(_("limit requires one to three arguments"))
1198 try:
1198 try:
1199 lim, ofs = 1, 0
1199 lim, ofs = 1, 0
1200 if 'n' in args:
1200 if 'n' in args:
1201 # i18n: "limit" is a keyword
1201 # i18n: "limit" is a keyword
1202 lim = int(getstring(args['n'], _("limit requires a number")))
1202 lim = int(getstring(args['n'], _("limit requires a number")))
1203 if 'offset' in args:
1203 if 'offset' in args:
1204 # i18n: "limit" is a keyword
1204 # i18n: "limit" is a keyword
1205 ofs = int(getstring(args['offset'], _("limit requires a number")))
1205 ofs = int(getstring(args['offset'], _("limit requires a number")))
1206 if ofs < 0:
1206 if ofs < 0:
1207 raise error.ParseError(_("negative offset"))
1207 raise error.ParseError(_("negative offset"))
1208 except (TypeError, ValueError):
1208 except (TypeError, ValueError):
1209 # i18n: "limit" is a keyword
1209 # i18n: "limit" is a keyword
1210 raise error.ParseError(_("limit expects a number"))
1210 raise error.ParseError(_("limit expects a number"))
1211 os = getset(repo, fullreposet(repo), args['set'])
1211 os = getset(repo, fullreposet(repo), args['set'])
1212 result = []
1212 result = []
1213 it = iter(os)
1213 it = iter(os)
1214 for x in xrange(ofs):
1214 for x in xrange(ofs):
1215 y = next(it, None)
1215 y = next(it, None)
1216 if y is None:
1216 if y is None:
1217 break
1217 break
1218 for x in xrange(lim):
1218 for x in xrange(lim):
1219 y = next(it, None)
1219 y = next(it, None)
1220 if y is None:
1220 if y is None:
1221 break
1221 break
1222 elif y in subset:
1222 elif y in subset:
1223 result.append(y)
1223 result.append(y)
1224 return baseset(result, datarepr=('<limit n=%d, offset=%d, %r, %r>',
1224 return baseset(result, datarepr=('<limit n=%d, offset=%d, %r, %r>',
1225 lim, ofs, subset, os))
1225 lim, ofs, subset, os))
1226
1226
1227 @predicate('last(set, [n])', safe=True)
1227 @predicate('last(set, [n])', safe=True)
1228 def last(repo, subset, x):
1228 def last(repo, subset, x):
1229 """Last n members of set, defaulting to 1.
1229 """Last n members of set, defaulting to 1.
1230 """
1230 """
1231 # i18n: "last" is a keyword
1231 # i18n: "last" is a keyword
1232 l = getargs(x, 1, 2, _("last requires one or two arguments"))
1232 l = getargs(x, 1, 2, _("last requires one or two arguments"))
1233 try:
1233 try:
1234 lim = 1
1234 lim = 1
1235 if len(l) == 2:
1235 if len(l) == 2:
1236 # i18n: "last" is a keyword
1236 # i18n: "last" is a keyword
1237 lim = int(getstring(l[1], _("last requires a number")))
1237 lim = int(getstring(l[1], _("last requires a number")))
1238 except (TypeError, ValueError):
1238 except (TypeError, ValueError):
1239 # i18n: "last" is a keyword
1239 # i18n: "last" is a keyword
1240 raise error.ParseError(_("last expects a number"))
1240 raise error.ParseError(_("last expects a number"))
1241 os = getset(repo, fullreposet(repo), l[0])
1241 os = getset(repo, fullreposet(repo), l[0])
1242 os.reverse()
1242 os.reverse()
1243 result = []
1243 result = []
1244 it = iter(os)
1244 it = iter(os)
1245 for x in xrange(lim):
1245 for x in xrange(lim):
1246 y = next(it, None)
1246 y = next(it, None)
1247 if y is None:
1247 if y is None:
1248 break
1248 break
1249 elif y in subset:
1249 elif y in subset:
1250 result.append(y)
1250 result.append(y)
1251 return baseset(result, datarepr=('<last n=%d, %r, %r>', lim, subset, os))
1251 return baseset(result, datarepr=('<last n=%d, %r, %r>', lim, subset, os))
1252
1252
1253 @predicate('max(set)', safe=True)
1253 @predicate('max(set)', safe=True)
1254 def maxrev(repo, subset, x):
1254 def maxrev(repo, subset, x):
1255 """Changeset with highest revision number in set.
1255 """Changeset with highest revision number in set.
1256 """
1256 """
1257 os = getset(repo, fullreposet(repo), x)
1257 os = getset(repo, fullreposet(repo), x)
1258 try:
1258 try:
1259 m = os.max()
1259 m = os.max()
1260 if m in subset:
1260 if m in subset:
1261 return baseset([m], datarepr=('<max %r, %r>', subset, os))
1261 return baseset([m], datarepr=('<max %r, %r>', subset, os))
1262 except ValueError:
1262 except ValueError:
1263 # os.max() throws a ValueError when the collection is empty.
1263 # os.max() throws a ValueError when the collection is empty.
1264 # Same as python's max().
1264 # Same as python's max().
1265 pass
1265 pass
1266 return baseset(datarepr=('<max %r, %r>', subset, os))
1266 return baseset(datarepr=('<max %r, %r>', subset, os))
1267
1267
1268 @predicate('merge()', safe=True)
1268 @predicate('merge()', safe=True)
1269 def merge(repo, subset, x):
1269 def merge(repo, subset, x):
1270 """Changeset is a merge changeset.
1270 """Changeset is a merge changeset.
1271 """
1271 """
1272 # i18n: "merge" is a keyword
1272 # i18n: "merge" is a keyword
1273 getargs(x, 0, 0, _("merge takes no arguments"))
1273 getargs(x, 0, 0, _("merge takes no arguments"))
1274 cl = repo.changelog
1274 cl = repo.changelog
1275 return subset.filter(lambda r: cl.parentrevs(r)[1] != -1,
1275 return subset.filter(lambda r: cl.parentrevs(r)[1] != -1,
1276 condrepr='<merge>')
1276 condrepr='<merge>')
1277
1277
1278 @predicate('branchpoint()', safe=True)
1278 @predicate('branchpoint()', safe=True)
1279 def branchpoint(repo, subset, x):
1279 def branchpoint(repo, subset, x):
1280 """Changesets with more than one child.
1280 """Changesets with more than one child.
1281 """
1281 """
1282 # i18n: "branchpoint" is a keyword
1282 # i18n: "branchpoint" is a keyword
1283 getargs(x, 0, 0, _("branchpoint takes no arguments"))
1283 getargs(x, 0, 0, _("branchpoint takes no arguments"))
1284 cl = repo.changelog
1284 cl = repo.changelog
1285 if not subset:
1285 if not subset:
1286 return baseset()
1286 return baseset()
1287 # XXX this should be 'parentset.min()' assuming 'parentset' is a smartset
1287 # XXX this should be 'parentset.min()' assuming 'parentset' is a smartset
1288 # (and if it is not, it should.)
1288 # (and if it is not, it should.)
1289 baserev = min(subset)
1289 baserev = min(subset)
1290 parentscount = [0]*(len(repo) - baserev)
1290 parentscount = [0]*(len(repo) - baserev)
1291 for r in cl.revs(start=baserev + 1):
1291 for r in cl.revs(start=baserev + 1):
1292 for p in cl.parentrevs(r):
1292 for p in cl.parentrevs(r):
1293 if p >= baserev:
1293 if p >= baserev:
1294 parentscount[p - baserev] += 1
1294 parentscount[p - baserev] += 1
1295 return subset.filter(lambda r: parentscount[r - baserev] > 1,
1295 return subset.filter(lambda r: parentscount[r - baserev] > 1,
1296 condrepr='<branchpoint>')
1296 condrepr='<branchpoint>')
1297
1297
1298 @predicate('min(set)', safe=True)
1298 @predicate('min(set)', safe=True)
1299 def minrev(repo, subset, x):
1299 def minrev(repo, subset, x):
1300 """Changeset with lowest revision number in set.
1300 """Changeset with lowest revision number in set.
1301 """
1301 """
1302 os = getset(repo, fullreposet(repo), x)
1302 os = getset(repo, fullreposet(repo), x)
1303 try:
1303 try:
1304 m = os.min()
1304 m = os.min()
1305 if m in subset:
1305 if m in subset:
1306 return baseset([m], datarepr=('<min %r, %r>', subset, os))
1306 return baseset([m], datarepr=('<min %r, %r>', subset, os))
1307 except ValueError:
1307 except ValueError:
1308 # os.min() throws a ValueError when the collection is empty.
1308 # os.min() throws a ValueError when the collection is empty.
1309 # Same as python's min().
1309 # Same as python's min().
1310 pass
1310 pass
1311 return baseset(datarepr=('<min %r, %r>', subset, os))
1311 return baseset(datarepr=('<min %r, %r>', subset, os))
1312
1312
1313 @predicate('modifies(pattern)', safe=True)
1313 @predicate('modifies(pattern)', safe=True)
1314 def modifies(repo, subset, x):
1314 def modifies(repo, subset, x):
1315 """Changesets modifying files matched by pattern.
1315 """Changesets modifying files matched by pattern.
1316
1316
1317 The pattern without explicit kind like ``glob:`` is expected to be
1317 The pattern without explicit kind like ``glob:`` is expected to be
1318 relative to the current directory and match against a file or a
1318 relative to the current directory and match against a file or a
1319 directory.
1319 directory.
1320 """
1320 """
1321 # i18n: "modifies" is a keyword
1321 # i18n: "modifies" is a keyword
1322 pat = getstring(x, _("modifies requires a pattern"))
1322 pat = getstring(x, _("modifies requires a pattern"))
1323 return checkstatus(repo, subset, pat, 0)
1323 return checkstatus(repo, subset, pat, 0)
1324
1324
1325 @predicate('named(namespace)')
1325 @predicate('named(namespace)')
1326 def named(repo, subset, x):
1326 def named(repo, subset, x):
1327 """The changesets in a given namespace.
1327 """The changesets in a given namespace.
1328
1328
1329 If `namespace` starts with `re:`, the remainder of the string is treated as
1329 If `namespace` starts with `re:`, the remainder of the string is treated as
1330 a regular expression. To match a namespace that actually starts with `re:`,
1330 a regular expression. To match a namespace that actually starts with `re:`,
1331 use the prefix `literal:`.
1331 use the prefix `literal:`.
1332 """
1332 """
1333 # i18n: "named" is a keyword
1333 # i18n: "named" is a keyword
1334 args = getargs(x, 1, 1, _('named requires a namespace argument'))
1334 args = getargs(x, 1, 1, _('named requires a namespace argument'))
1335
1335
1336 ns = getstring(args[0],
1336 ns = getstring(args[0],
1337 # i18n: "named" is a keyword
1337 # i18n: "named" is a keyword
1338 _('the argument to named must be a string'))
1338 _('the argument to named must be a string'))
1339 kind, pattern, matcher = util.stringmatcher(ns)
1339 kind, pattern, matcher = util.stringmatcher(ns)
1340 namespaces = set()
1340 namespaces = set()
1341 if kind == 'literal':
1341 if kind == 'literal':
1342 if pattern not in repo.names:
1342 if pattern not in repo.names:
1343 raise error.RepoLookupError(_("namespace '%s' does not exist")
1343 raise error.RepoLookupError(_("namespace '%s' does not exist")
1344 % ns)
1344 % ns)
1345 namespaces.add(repo.names[pattern])
1345 namespaces.add(repo.names[pattern])
1346 else:
1346 else:
1347 for name, ns in repo.names.iteritems():
1347 for name, ns in repo.names.iteritems():
1348 if matcher(name):
1348 if matcher(name):
1349 namespaces.add(ns)
1349 namespaces.add(ns)
1350 if not namespaces:
1350 if not namespaces:
1351 raise error.RepoLookupError(_("no namespace exists"
1351 raise error.RepoLookupError(_("no namespace exists"
1352 " that match '%s'") % pattern)
1352 " that match '%s'") % pattern)
1353
1353
1354 names = set()
1354 names = set()
1355 for ns in namespaces:
1355 for ns in namespaces:
1356 for name in ns.listnames(repo):
1356 for name in ns.listnames(repo):
1357 if name not in ns.deprecated:
1357 if name not in ns.deprecated:
1358 names.update(repo[n].rev() for n in ns.nodes(repo, name))
1358 names.update(repo[n].rev() for n in ns.nodes(repo, name))
1359
1359
1360 names -= set([node.nullrev])
1360 names -= set([node.nullrev])
1361 return subset & names
1361 return subset & names
1362
1362
1363 @predicate('id(string)', safe=True)
1363 @predicate('id(string)', safe=True)
1364 def node_(repo, subset, x):
1364 def node_(repo, subset, x):
1365 """Revision non-ambiguously specified by the given hex string prefix.
1365 """Revision non-ambiguously specified by the given hex string prefix.
1366 """
1366 """
1367 # i18n: "id" is a keyword
1367 # i18n: "id" is a keyword
1368 l = getargs(x, 1, 1, _("id requires one argument"))
1368 l = getargs(x, 1, 1, _("id requires one argument"))
1369 # i18n: "id" is a keyword
1369 # i18n: "id" is a keyword
1370 n = getstring(l[0], _("id requires a string"))
1370 n = getstring(l[0], _("id requires a string"))
1371 if len(n) == 40:
1371 if len(n) == 40:
1372 try:
1372 try:
1373 rn = repo.changelog.rev(node.bin(n))
1373 rn = repo.changelog.rev(node.bin(n))
1374 except (LookupError, TypeError):
1374 except (LookupError, TypeError):
1375 rn = None
1375 rn = None
1376 else:
1376 else:
1377 rn = None
1377 rn = None
1378 pm = repo.changelog._partialmatch(n)
1378 pm = repo.changelog._partialmatch(n)
1379 if pm is not None:
1379 if pm is not None:
1380 rn = repo.changelog.rev(pm)
1380 rn = repo.changelog.rev(pm)
1381
1381
1382 if rn is None:
1382 if rn is None:
1383 return baseset()
1383 return baseset()
1384 result = baseset([rn])
1384 result = baseset([rn])
1385 return result & subset
1385 return result & subset
1386
1386
1387 @predicate('obsolete()', safe=True)
1387 @predicate('obsolete()', safe=True)
1388 def obsolete(repo, subset, x):
1388 def obsolete(repo, subset, x):
1389 """Mutable changeset with a newer version."""
1389 """Mutable changeset with a newer version."""
1390 # i18n: "obsolete" is a keyword
1390 # i18n: "obsolete" is a keyword
1391 getargs(x, 0, 0, _("obsolete takes no arguments"))
1391 getargs(x, 0, 0, _("obsolete takes no arguments"))
1392 obsoletes = obsmod.getrevs(repo, 'obsolete')
1392 obsoletes = obsmod.getrevs(repo, 'obsolete')
1393 return subset & obsoletes
1393 return subset & obsoletes
1394
1394
1395 @predicate('only(set, [set])', safe=True)
1395 @predicate('only(set, [set])', safe=True)
1396 def only(repo, subset, x):
1396 def only(repo, subset, x):
1397 """Changesets that are ancestors of the first set that are not ancestors
1397 """Changesets that are ancestors of the first set that are not ancestors
1398 of any other head in the repo. If a second set is specified, the result
1398 of any other head in the repo. If a second set is specified, the result
1399 is ancestors of the first set that are not ancestors of the second set
1399 is ancestors of the first set that are not ancestors of the second set
1400 (i.e. ::<set1> - ::<set2>).
1400 (i.e. ::<set1> - ::<set2>).
1401 """
1401 """
1402 cl = repo.changelog
1402 cl = repo.changelog
1403 # i18n: "only" is a keyword
1403 # i18n: "only" is a keyword
1404 args = getargs(x, 1, 2, _('only takes one or two arguments'))
1404 args = getargs(x, 1, 2, _('only takes one or two arguments'))
1405 include = getset(repo, fullreposet(repo), args[0])
1405 include = getset(repo, fullreposet(repo), args[0])
1406 if len(args) == 1:
1406 if len(args) == 1:
1407 if not include:
1407 if not include:
1408 return baseset()
1408 return baseset()
1409
1409
1410 descendants = set(_revdescendants(repo, include, False))
1410 descendants = set(_revdescendants(repo, include, False))
1411 exclude = [rev for rev in cl.headrevs()
1411 exclude = [rev for rev in cl.headrevs()
1412 if not rev in descendants and not rev in include]
1412 if not rev in descendants and not rev in include]
1413 else:
1413 else:
1414 exclude = getset(repo, fullreposet(repo), args[1])
1414 exclude = getset(repo, fullreposet(repo), args[1])
1415
1415
1416 results = set(cl.findmissingrevs(common=exclude, heads=include))
1416 results = set(cl.findmissingrevs(common=exclude, heads=include))
1417 # XXX we should turn this into a baseset instead of a set, smartset may do
1417 # XXX we should turn this into a baseset instead of a set, smartset may do
1418 # some optimisations from the fact this is a baseset.
1418 # some optimisations from the fact this is a baseset.
1419 return subset & results
1419 return subset & results
1420
1420
1421 @predicate('origin([set])', safe=True)
1421 @predicate('origin([set])', safe=True)
1422 def origin(repo, subset, x):
1422 def origin(repo, subset, x):
1423 """
1423 """
1424 Changesets that were specified as a source for the grafts, transplants or
1424 Changesets that were specified as a source for the grafts, transplants or
1425 rebases that created the given revisions. Omitting the optional set is the
1425 rebases that created the given revisions. Omitting the optional set is the
1426 same as passing all(). If a changeset created by these operations is itself
1426 same as passing all(). If a changeset created by these operations is itself
1427 specified as a source for one of these operations, only the source changeset
1427 specified as a source for one of these operations, only the source changeset
1428 for the first operation is selected.
1428 for the first operation is selected.
1429 """
1429 """
1430 if x is not None:
1430 if x is not None:
1431 dests = getset(repo, fullreposet(repo), x)
1431 dests = getset(repo, fullreposet(repo), x)
1432 else:
1432 else:
1433 dests = fullreposet(repo)
1433 dests = fullreposet(repo)
1434
1434
1435 def _firstsrc(rev):
1435 def _firstsrc(rev):
1436 src = _getrevsource(repo, rev)
1436 src = _getrevsource(repo, rev)
1437 if src is None:
1437 if src is None:
1438 return None
1438 return None
1439
1439
1440 while True:
1440 while True:
1441 prev = _getrevsource(repo, src)
1441 prev = _getrevsource(repo, src)
1442
1442
1443 if prev is None:
1443 if prev is None:
1444 return src
1444 return src
1445 src = prev
1445 src = prev
1446
1446
1447 o = set([_firstsrc(r) for r in dests])
1447 o = set([_firstsrc(r) for r in dests])
1448 o -= set([None])
1448 o -= set([None])
1449 # XXX we should turn this into a baseset instead of a set, smartset may do
1449 # XXX we should turn this into a baseset instead of a set, smartset may do
1450 # some optimisations from the fact this is a baseset.
1450 # some optimisations from the fact this is a baseset.
1451 return subset & o
1451 return subset & o
1452
1452
1453 @predicate('outgoing([path])', safe=True)
1453 @predicate('outgoing([path])', safe=True)
1454 def outgoing(repo, subset, x):
1454 def outgoing(repo, subset, x):
1455 """Changesets not found in the specified destination repository, or the
1455 """Changesets not found in the specified destination repository, or the
1456 default push location.
1456 default push location.
1457 """
1457 """
1458 # Avoid cycles.
1458 # Avoid cycles.
1459 from . import (
1459 from . import (
1460 discovery,
1460 discovery,
1461 hg,
1461 hg,
1462 )
1462 )
1463 # i18n: "outgoing" is a keyword
1463 # i18n: "outgoing" is a keyword
1464 l = getargs(x, 0, 1, _("outgoing takes one or no arguments"))
1464 l = getargs(x, 0, 1, _("outgoing takes one or no arguments"))
1465 # i18n: "outgoing" is a keyword
1465 # i18n: "outgoing" is a keyword
1466 dest = l and getstring(l[0], _("outgoing requires a repository path")) or ''
1466 dest = l and getstring(l[0], _("outgoing requires a repository path")) or ''
1467 dest = repo.ui.expandpath(dest or 'default-push', dest or 'default')
1467 dest = repo.ui.expandpath(dest or 'default-push', dest or 'default')
1468 dest, branches = hg.parseurl(dest)
1468 dest, branches = hg.parseurl(dest)
1469 revs, checkout = hg.addbranchrevs(repo, repo, branches, [])
1469 revs, checkout = hg.addbranchrevs(repo, repo, branches, [])
1470 if revs:
1470 if revs:
1471 revs = [repo.lookup(rev) for rev in revs]
1471 revs = [repo.lookup(rev) for rev in revs]
1472 other = hg.peer(repo, {}, dest)
1472 other = hg.peer(repo, {}, dest)
1473 repo.ui.pushbuffer()
1473 repo.ui.pushbuffer()
1474 outgoing = discovery.findcommonoutgoing(repo, other, onlyheads=revs)
1474 outgoing = discovery.findcommonoutgoing(repo, other, onlyheads=revs)
1475 repo.ui.popbuffer()
1475 repo.ui.popbuffer()
1476 cl = repo.changelog
1476 cl = repo.changelog
1477 o = set([cl.rev(r) for r in outgoing.missing])
1477 o = set([cl.rev(r) for r in outgoing.missing])
1478 return subset & o
1478 return subset & o
1479
1479
1480 @predicate('p1([set])', safe=True)
1480 @predicate('p1([set])', safe=True)
1481 def p1(repo, subset, x):
1481 def p1(repo, subset, x):
1482 """First parent of changesets in set, or the working directory.
1482 """First parent of changesets in set, or the working directory.
1483 """
1483 """
1484 if x is None:
1484 if x is None:
1485 p = repo[x].p1().rev()
1485 p = repo[x].p1().rev()
1486 if p >= 0:
1486 if p >= 0:
1487 return subset & baseset([p])
1487 return subset & baseset([p])
1488 return baseset()
1488 return baseset()
1489
1489
1490 ps = set()
1490 ps = set()
1491 cl = repo.changelog
1491 cl = repo.changelog
1492 for r in getset(repo, fullreposet(repo), x):
1492 for r in getset(repo, fullreposet(repo), x):
1493 ps.add(cl.parentrevs(r)[0])
1493 ps.add(cl.parentrevs(r)[0])
1494 ps -= set([node.nullrev])
1494 ps -= set([node.nullrev])
1495 # XXX we should turn this into a baseset instead of a set, smartset may do
1495 # XXX we should turn this into a baseset instead of a set, smartset may do
1496 # some optimisations from the fact this is a baseset.
1496 # some optimisations from the fact this is a baseset.
1497 return subset & ps
1497 return subset & ps
1498
1498
1499 @predicate('p2([set])', safe=True)
1499 @predicate('p2([set])', safe=True)
1500 def p2(repo, subset, x):
1500 def p2(repo, subset, x):
1501 """Second parent of changesets in set, or the working directory.
1501 """Second parent of changesets in set, or the working directory.
1502 """
1502 """
1503 if x is None:
1503 if x is None:
1504 ps = repo[x].parents()
1504 ps = repo[x].parents()
1505 try:
1505 try:
1506 p = ps[1].rev()
1506 p = ps[1].rev()
1507 if p >= 0:
1507 if p >= 0:
1508 return subset & baseset([p])
1508 return subset & baseset([p])
1509 return baseset()
1509 return baseset()
1510 except IndexError:
1510 except IndexError:
1511 return baseset()
1511 return baseset()
1512
1512
1513 ps = set()
1513 ps = set()
1514 cl = repo.changelog
1514 cl = repo.changelog
1515 for r in getset(repo, fullreposet(repo), x):
1515 for r in getset(repo, fullreposet(repo), x):
1516 ps.add(cl.parentrevs(r)[1])
1516 ps.add(cl.parentrevs(r)[1])
1517 ps -= set([node.nullrev])
1517 ps -= set([node.nullrev])
1518 # XXX we should turn this into a baseset instead of a set, smartset may do
1518 # XXX we should turn this into a baseset instead of a set, smartset may do
1519 # some optimisations from the fact this is a baseset.
1519 # some optimisations from the fact this is a baseset.
1520 return subset & ps
1520 return subset & ps
1521
1521
1522 @predicate('parents([set])', safe=True)
1522 @predicate('parents([set])', safe=True)
1523 def parents(repo, subset, x):
1523 def parents(repo, subset, x):
1524 """
1524 """
1525 The set of all parents for all changesets in set, or the working directory.
1525 The set of all parents for all changesets in set, or the working directory.
1526 """
1526 """
1527 if x is None:
1527 if x is None:
1528 ps = set(p.rev() for p in repo[x].parents())
1528 ps = set(p.rev() for p in repo[x].parents())
1529 else:
1529 else:
1530 ps = set()
1530 ps = set()
1531 cl = repo.changelog
1531 cl = repo.changelog
1532 up = ps.update
1532 up = ps.update
1533 parentrevs = cl.parentrevs
1533 parentrevs = cl.parentrevs
1534 for r in getset(repo, fullreposet(repo), x):
1534 for r in getset(repo, fullreposet(repo), x):
1535 if r == node.wdirrev:
1535 if r == node.wdirrev:
1536 up(p.rev() for p in repo[r].parents())
1536 up(p.rev() for p in repo[r].parents())
1537 else:
1537 else:
1538 up(parentrevs(r))
1538 up(parentrevs(r))
1539 ps -= set([node.nullrev])
1539 ps -= set([node.nullrev])
1540 return subset & ps
1540 return subset & ps
1541
1541
1542 def _phase(repo, subset, target):
1542 def _phase(repo, subset, target):
1543 """helper to select all rev in phase <target>"""
1543 """helper to select all rev in phase <target>"""
1544 repo._phasecache.loadphaserevs(repo) # ensure phase's sets are loaded
1544 repo._phasecache.loadphaserevs(repo) # ensure phase's sets are loaded
1545 if repo._phasecache._phasesets:
1545 if repo._phasecache._phasesets:
1546 s = repo._phasecache._phasesets[target] - repo.changelog.filteredrevs
1546 s = repo._phasecache._phasesets[target] - repo.changelog.filteredrevs
1547 s = baseset(s)
1547 s = baseset(s)
1548 s.sort() # set are non ordered, so we enforce ascending
1548 s.sort() # set are non ordered, so we enforce ascending
1549 return subset & s
1549 return subset & s
1550 else:
1550 else:
1551 phase = repo._phasecache.phase
1551 phase = repo._phasecache.phase
1552 condition = lambda r: phase(repo, r) == target
1552 condition = lambda r: phase(repo, r) == target
1553 return subset.filter(condition, condrepr=('<phase %r>', target),
1553 return subset.filter(condition, condrepr=('<phase %r>', target),
1554 cache=False)
1554 cache=False)
1555
1555
1556 @predicate('draft()', safe=True)
1556 @predicate('draft()', safe=True)
1557 def draft(repo, subset, x):
1557 def draft(repo, subset, x):
1558 """Changeset in draft phase."""
1558 """Changeset in draft phase."""
1559 # i18n: "draft" is a keyword
1559 # i18n: "draft" is a keyword
1560 getargs(x, 0, 0, _("draft takes no arguments"))
1560 getargs(x, 0, 0, _("draft takes no arguments"))
1561 target = phases.draft
1561 target = phases.draft
1562 return _phase(repo, subset, target)
1562 return _phase(repo, subset, target)
1563
1563
1564 @predicate('secret()', safe=True)
1564 @predicate('secret()', safe=True)
1565 def secret(repo, subset, x):
1565 def secret(repo, subset, x):
1566 """Changeset in secret phase."""
1566 """Changeset in secret phase."""
1567 # i18n: "secret" is a keyword
1567 # i18n: "secret" is a keyword
1568 getargs(x, 0, 0, _("secret takes no arguments"))
1568 getargs(x, 0, 0, _("secret takes no arguments"))
1569 target = phases.secret
1569 target = phases.secret
1570 return _phase(repo, subset, target)
1570 return _phase(repo, subset, target)
1571
1571
1572 def parentspec(repo, subset, x, n):
1572 def parentspec(repo, subset, x, n):
1573 """``set^0``
1573 """``set^0``
1574 The set.
1574 The set.
1575 ``set^1`` (or ``set^``), ``set^2``
1575 ``set^1`` (or ``set^``), ``set^2``
1576 First or second parent, respectively, of all changesets in set.
1576 First or second parent, respectively, of all changesets in set.
1577 """
1577 """
1578 try:
1578 try:
1579 n = int(n[1])
1579 n = int(n[1])
1580 if n not in (0, 1, 2):
1580 if n not in (0, 1, 2):
1581 raise ValueError
1581 raise ValueError
1582 except (TypeError, ValueError):
1582 except (TypeError, ValueError):
1583 raise error.ParseError(_("^ expects a number 0, 1, or 2"))
1583 raise error.ParseError(_("^ expects a number 0, 1, or 2"))
1584 ps = set()
1584 ps = set()
1585 cl = repo.changelog
1585 cl = repo.changelog
1586 for r in getset(repo, fullreposet(repo), x):
1586 for r in getset(repo, fullreposet(repo), x):
1587 if n == 0:
1587 if n == 0:
1588 ps.add(r)
1588 ps.add(r)
1589 elif n == 1:
1589 elif n == 1:
1590 ps.add(cl.parentrevs(r)[0])
1590 ps.add(cl.parentrevs(r)[0])
1591 elif n == 2:
1591 elif n == 2:
1592 parents = cl.parentrevs(r)
1592 parents = cl.parentrevs(r)
1593 if len(parents) > 1:
1593 if len(parents) > 1:
1594 ps.add(parents[1])
1594 ps.add(parents[1])
1595 return subset & ps
1595 return subset & ps
1596
1596
1597 @predicate('present(set)', safe=True)
1597 @predicate('present(set)', safe=True)
1598 def present(repo, subset, x):
1598 def present(repo, subset, x):
1599 """An empty set, if any revision in set isn't found; otherwise,
1599 """An empty set, if any revision in set isn't found; otherwise,
1600 all revisions in set.
1600 all revisions in set.
1601
1601
1602 If any of specified revisions is not present in the local repository,
1602 If any of specified revisions is not present in the local repository,
1603 the query is normally aborted. But this predicate allows the query
1603 the query is normally aborted. But this predicate allows the query
1604 to continue even in such cases.
1604 to continue even in such cases.
1605 """
1605 """
1606 try:
1606 try:
1607 return getset(repo, subset, x)
1607 return getset(repo, subset, x)
1608 except error.RepoLookupError:
1608 except error.RepoLookupError:
1609 return baseset()
1609 return baseset()
1610
1610
1611 # for internal use
1611 # for internal use
1612 @predicate('_notpublic', safe=True)
1612 @predicate('_notpublic', safe=True)
1613 def _notpublic(repo, subset, x):
1613 def _notpublic(repo, subset, x):
1614 getargs(x, 0, 0, "_notpublic takes no arguments")
1614 getargs(x, 0, 0, "_notpublic takes no arguments")
1615 repo._phasecache.loadphaserevs(repo) # ensure phase's sets are loaded
1615 repo._phasecache.loadphaserevs(repo) # ensure phase's sets are loaded
1616 if repo._phasecache._phasesets:
1616 if repo._phasecache._phasesets:
1617 s = set()
1617 s = set()
1618 for u in repo._phasecache._phasesets[1:]:
1618 for u in repo._phasecache._phasesets[1:]:
1619 s.update(u)
1619 s.update(u)
1620 s = baseset(s - repo.changelog.filteredrevs)
1620 s = baseset(s - repo.changelog.filteredrevs)
1621 s.sort()
1621 s.sort()
1622 return subset & s
1622 return subset & s
1623 else:
1623 else:
1624 phase = repo._phasecache.phase
1624 phase = repo._phasecache.phase
1625 target = phases.public
1625 target = phases.public
1626 condition = lambda r: phase(repo, r) != target
1626 condition = lambda r: phase(repo, r) != target
1627 return subset.filter(condition, condrepr=('<phase %r>', target),
1627 return subset.filter(condition, condrepr=('<phase %r>', target),
1628 cache=False)
1628 cache=False)
1629
1629
1630 @predicate('public()', safe=True)
1630 @predicate('public()', safe=True)
1631 def public(repo, subset, x):
1631 def public(repo, subset, x):
1632 """Changeset in public phase."""
1632 """Changeset in public phase."""
1633 # i18n: "public" is a keyword
1633 # i18n: "public" is a keyword
1634 getargs(x, 0, 0, _("public takes no arguments"))
1634 getargs(x, 0, 0, _("public takes no arguments"))
1635 phase = repo._phasecache.phase
1635 phase = repo._phasecache.phase
1636 target = phases.public
1636 target = phases.public
1637 condition = lambda r: phase(repo, r) == target
1637 condition = lambda r: phase(repo, r) == target
1638 return subset.filter(condition, condrepr=('<phase %r>', target),
1638 return subset.filter(condition, condrepr=('<phase %r>', target),
1639 cache=False)
1639 cache=False)
1640
1640
1641 @predicate('remote([id [,path]])', safe=True)
1641 @predicate('remote([id [,path]])', safe=True)
1642 def remote(repo, subset, x):
1642 def remote(repo, subset, x):
1643 """Local revision that corresponds to the given identifier in a
1643 """Local revision that corresponds to the given identifier in a
1644 remote repository, if present. Here, the '.' identifier is a
1644 remote repository, if present. Here, the '.' identifier is a
1645 synonym for the current local branch.
1645 synonym for the current local branch.
1646 """
1646 """
1647
1647
1648 from . import hg # avoid start-up nasties
1648 from . import hg # avoid start-up nasties
1649 # i18n: "remote" is a keyword
1649 # i18n: "remote" is a keyword
1650 l = getargs(x, 0, 2, _("remote takes zero, one, or two arguments"))
1650 l = getargs(x, 0, 2, _("remote takes zero, one, or two arguments"))
1651
1651
1652 q = '.'
1652 q = '.'
1653 if len(l) > 0:
1653 if len(l) > 0:
1654 # i18n: "remote" is a keyword
1654 # i18n: "remote" is a keyword
1655 q = getstring(l[0], _("remote requires a string id"))
1655 q = getstring(l[0], _("remote requires a string id"))
1656 if q == '.':
1656 if q == '.':
1657 q = repo['.'].branch()
1657 q = repo['.'].branch()
1658
1658
1659 dest = ''
1659 dest = ''
1660 if len(l) > 1:
1660 if len(l) > 1:
1661 # i18n: "remote" is a keyword
1661 # i18n: "remote" is a keyword
1662 dest = getstring(l[1], _("remote requires a repository path"))
1662 dest = getstring(l[1], _("remote requires a repository path"))
1663 dest = repo.ui.expandpath(dest or 'default')
1663 dest = repo.ui.expandpath(dest or 'default')
1664 dest, branches = hg.parseurl(dest)
1664 dest, branches = hg.parseurl(dest)
1665 revs, checkout = hg.addbranchrevs(repo, repo, branches, [])
1665 revs, checkout = hg.addbranchrevs(repo, repo, branches, [])
1666 if revs:
1666 if revs:
1667 revs = [repo.lookup(rev) for rev in revs]
1667 revs = [repo.lookup(rev) for rev in revs]
1668 other = hg.peer(repo, {}, dest)
1668 other = hg.peer(repo, {}, dest)
1669 n = other.lookup(q)
1669 n = other.lookup(q)
1670 if n in repo:
1670 if n in repo:
1671 r = repo[n].rev()
1671 r = repo[n].rev()
1672 if r in subset:
1672 if r in subset:
1673 return baseset([r])
1673 return baseset([r])
1674 return baseset()
1674 return baseset()
1675
1675
1676 @predicate('removes(pattern)', safe=True)
1676 @predicate('removes(pattern)', safe=True)
1677 def removes(repo, subset, x):
1677 def removes(repo, subset, x):
1678 """Changesets which remove files matching pattern.
1678 """Changesets which remove files matching pattern.
1679
1679
1680 The pattern without explicit kind like ``glob:`` is expected to be
1680 The pattern without explicit kind like ``glob:`` is expected to be
1681 relative to the current directory and match against a file or a
1681 relative to the current directory and match against a file or a
1682 directory.
1682 directory.
1683 """
1683 """
1684 # i18n: "removes" is a keyword
1684 # i18n: "removes" is a keyword
1685 pat = getstring(x, _("removes requires a pattern"))
1685 pat = getstring(x, _("removes requires a pattern"))
1686 return checkstatus(repo, subset, pat, 2)
1686 return checkstatus(repo, subset, pat, 2)
1687
1687
1688 @predicate('rev(number)', safe=True)
1688 @predicate('rev(number)', safe=True)
1689 def rev(repo, subset, x):
1689 def rev(repo, subset, x):
1690 """Revision with the given numeric identifier.
1690 """Revision with the given numeric identifier.
1691 """
1691 """
1692 # i18n: "rev" is a keyword
1692 # i18n: "rev" is a keyword
1693 l = getargs(x, 1, 1, _("rev requires one argument"))
1693 l = getargs(x, 1, 1, _("rev requires one argument"))
1694 try:
1694 try:
1695 # i18n: "rev" is a keyword
1695 # i18n: "rev" is a keyword
1696 l = int(getstring(l[0], _("rev requires a number")))
1696 l = int(getstring(l[0], _("rev requires a number")))
1697 except (TypeError, ValueError):
1697 except (TypeError, ValueError):
1698 # i18n: "rev" is a keyword
1698 # i18n: "rev" is a keyword
1699 raise error.ParseError(_("rev expects a number"))
1699 raise error.ParseError(_("rev expects a number"))
1700 if l not in repo.changelog and l != node.nullrev:
1700 if l not in repo.changelog and l != node.nullrev:
1701 return baseset()
1701 return baseset()
1702 return subset & baseset([l])
1702 return subset & baseset([l])
1703
1703
1704 @predicate('matching(revision [, field])', safe=True)
1704 @predicate('matching(revision [, field])', safe=True)
1705 def matching(repo, subset, x):
1705 def matching(repo, subset, x):
1706 """Changesets in which a given set of fields match the set of fields in the
1706 """Changesets in which a given set of fields match the set of fields in the
1707 selected revision or set.
1707 selected revision or set.
1708
1708
1709 To match more than one field pass the list of fields to match separated
1709 To match more than one field pass the list of fields to match separated
1710 by spaces (e.g. ``author description``).
1710 by spaces (e.g. ``author description``).
1711
1711
1712 Valid fields are most regular revision fields and some special fields.
1712 Valid fields are most regular revision fields and some special fields.
1713
1713
1714 Regular revision fields are ``description``, ``author``, ``branch``,
1714 Regular revision fields are ``description``, ``author``, ``branch``,
1715 ``date``, ``files``, ``phase``, ``parents``, ``substate``, ``user``
1715 ``date``, ``files``, ``phase``, ``parents``, ``substate``, ``user``
1716 and ``diff``.
1716 and ``diff``.
1717 Note that ``author`` and ``user`` are synonyms. ``diff`` refers to the
1717 Note that ``author`` and ``user`` are synonyms. ``diff`` refers to the
1718 contents of the revision. Two revisions matching their ``diff`` will
1718 contents of the revision. Two revisions matching their ``diff`` will
1719 also match their ``files``.
1719 also match their ``files``.
1720
1720
1721 Special fields are ``summary`` and ``metadata``:
1721 Special fields are ``summary`` and ``metadata``:
1722 ``summary`` matches the first line of the description.
1722 ``summary`` matches the first line of the description.
1723 ``metadata`` is equivalent to matching ``description user date``
1723 ``metadata`` is equivalent to matching ``description user date``
1724 (i.e. it matches the main metadata fields).
1724 (i.e. it matches the main metadata fields).
1725
1725
1726 ``metadata`` is the default field which is used when no fields are
1726 ``metadata`` is the default field which is used when no fields are
1727 specified. You can match more than one field at a time.
1727 specified. You can match more than one field at a time.
1728 """
1728 """
1729 # i18n: "matching" is a keyword
1729 # i18n: "matching" is a keyword
1730 l = getargs(x, 1, 2, _("matching takes 1 or 2 arguments"))
1730 l = getargs(x, 1, 2, _("matching takes 1 or 2 arguments"))
1731
1731
1732 revs = getset(repo, fullreposet(repo), l[0])
1732 revs = getset(repo, fullreposet(repo), l[0])
1733
1733
1734 fieldlist = ['metadata']
1734 fieldlist = ['metadata']
1735 if len(l) > 1:
1735 if len(l) > 1:
1736 fieldlist = getstring(l[1],
1736 fieldlist = getstring(l[1],
1737 # i18n: "matching" is a keyword
1737 # i18n: "matching" is a keyword
1738 _("matching requires a string "
1738 _("matching requires a string "
1739 "as its second argument")).split()
1739 "as its second argument")).split()
1740
1740
1741 # Make sure that there are no repeated fields,
1741 # Make sure that there are no repeated fields,
1742 # expand the 'special' 'metadata' field type
1742 # expand the 'special' 'metadata' field type
1743 # and check the 'files' whenever we check the 'diff'
1743 # and check the 'files' whenever we check the 'diff'
1744 fields = []
1744 fields = []
1745 for field in fieldlist:
1745 for field in fieldlist:
1746 if field == 'metadata':
1746 if field == 'metadata':
1747 fields += ['user', 'description', 'date']
1747 fields += ['user', 'description', 'date']
1748 elif field == 'diff':
1748 elif field == 'diff':
1749 # a revision matching the diff must also match the files
1749 # a revision matching the diff must also match the files
1750 # since matching the diff is very costly, make sure to
1750 # since matching the diff is very costly, make sure to
1751 # also match the files first
1751 # also match the files first
1752 fields += ['files', 'diff']
1752 fields += ['files', 'diff']
1753 else:
1753 else:
1754 if field == 'author':
1754 if field == 'author':
1755 field = 'user'
1755 field = 'user'
1756 fields.append(field)
1756 fields.append(field)
1757 fields = set(fields)
1757 fields = set(fields)
1758 if 'summary' in fields and 'description' in fields:
1758 if 'summary' in fields and 'description' in fields:
1759 # If a revision matches its description it also matches its summary
1759 # If a revision matches its description it also matches its summary
1760 fields.discard('summary')
1760 fields.discard('summary')
1761
1761
1762 # We may want to match more than one field
1762 # We may want to match more than one field
1763 # Not all fields take the same amount of time to be matched
1763 # Not all fields take the same amount of time to be matched
1764 # Sort the selected fields in order of increasing matching cost
1764 # Sort the selected fields in order of increasing matching cost
1765 fieldorder = ['phase', 'parents', 'user', 'date', 'branch', 'summary',
1765 fieldorder = ['phase', 'parents', 'user', 'date', 'branch', 'summary',
1766 'files', 'description', 'substate', 'diff']
1766 'files', 'description', 'substate', 'diff']
1767 def fieldkeyfunc(f):
1767 def fieldkeyfunc(f):
1768 try:
1768 try:
1769 return fieldorder.index(f)
1769 return fieldorder.index(f)
1770 except ValueError:
1770 except ValueError:
1771 # assume an unknown field is very costly
1771 # assume an unknown field is very costly
1772 return len(fieldorder)
1772 return len(fieldorder)
1773 fields = list(fields)
1773 fields = list(fields)
1774 fields.sort(key=fieldkeyfunc)
1774 fields.sort(key=fieldkeyfunc)
1775
1775
1776 # Each field will be matched with its own "getfield" function
1776 # Each field will be matched with its own "getfield" function
1777 # which will be added to the getfieldfuncs array of functions
1777 # which will be added to the getfieldfuncs array of functions
1778 getfieldfuncs = []
1778 getfieldfuncs = []
1779 _funcs = {
1779 _funcs = {
1780 'user': lambda r: repo[r].user(),
1780 'user': lambda r: repo[r].user(),
1781 'branch': lambda r: repo[r].branch(),
1781 'branch': lambda r: repo[r].branch(),
1782 'date': lambda r: repo[r].date(),
1782 'date': lambda r: repo[r].date(),
1783 'description': lambda r: repo[r].description(),
1783 'description': lambda r: repo[r].description(),
1784 'files': lambda r: repo[r].files(),
1784 'files': lambda r: repo[r].files(),
1785 'parents': lambda r: repo[r].parents(),
1785 'parents': lambda r: repo[r].parents(),
1786 'phase': lambda r: repo[r].phase(),
1786 'phase': lambda r: repo[r].phase(),
1787 'substate': lambda r: repo[r].substate,
1787 'substate': lambda r: repo[r].substate,
1788 'summary': lambda r: repo[r].description().splitlines()[0],
1788 'summary': lambda r: repo[r].description().splitlines()[0],
1789 'diff': lambda r: list(repo[r].diff(git=True),)
1789 'diff': lambda r: list(repo[r].diff(git=True),)
1790 }
1790 }
1791 for info in fields:
1791 for info in fields:
1792 getfield = _funcs.get(info, None)
1792 getfield = _funcs.get(info, None)
1793 if getfield is None:
1793 if getfield is None:
1794 raise error.ParseError(
1794 raise error.ParseError(
1795 # i18n: "matching" is a keyword
1795 # i18n: "matching" is a keyword
1796 _("unexpected field name passed to matching: %s") % info)
1796 _("unexpected field name passed to matching: %s") % info)
1797 getfieldfuncs.append(getfield)
1797 getfieldfuncs.append(getfield)
1798 # convert the getfield array of functions into a "getinfo" function
1798 # convert the getfield array of functions into a "getinfo" function
1799 # which returns an array of field values (or a single value if there
1799 # which returns an array of field values (or a single value if there
1800 # is only one field to match)
1800 # is only one field to match)
1801 getinfo = lambda r: [f(r) for f in getfieldfuncs]
1801 getinfo = lambda r: [f(r) for f in getfieldfuncs]
1802
1802
1803 def matches(x):
1803 def matches(x):
1804 for rev in revs:
1804 for rev in revs:
1805 target = getinfo(rev)
1805 target = getinfo(rev)
1806 match = True
1806 match = True
1807 for n, f in enumerate(getfieldfuncs):
1807 for n, f in enumerate(getfieldfuncs):
1808 if target[n] != f(x):
1808 if target[n] != f(x):
1809 match = False
1809 match = False
1810 if match:
1810 if match:
1811 return True
1811 return True
1812 return False
1812 return False
1813
1813
1814 return subset.filter(matches, condrepr=('<matching%r %r>', fields, revs))
1814 return subset.filter(matches, condrepr=('<matching%r %r>', fields, revs))
1815
1815
1816 @predicate('reverse(set)', safe=True)
1816 @predicate('reverse(set)', safe=True)
1817 def reverse(repo, subset, x):
1817 def reverse(repo, subset, x):
1818 """Reverse order of set.
1818 """Reverse order of set.
1819 """
1819 """
1820 l = getset(repo, subset, x)
1820 l = getset(repo, subset, x)
1821 l.reverse()
1821 l.reverse()
1822 return l
1822 return l
1823
1823
1824 @predicate('roots(set)', safe=True)
1824 @predicate('roots(set)', safe=True)
1825 def roots(repo, subset, x):
1825 def roots(repo, subset, x):
1826 """Changesets in set with no parent changeset in set.
1826 """Changesets in set with no parent changeset in set.
1827 """
1827 """
1828 s = getset(repo, fullreposet(repo), x)
1828 s = getset(repo, fullreposet(repo), x)
1829 parents = repo.changelog.parentrevs
1829 parents = repo.changelog.parentrevs
1830 def filter(r):
1830 def filter(r):
1831 for p in parents(r):
1831 for p in parents(r):
1832 if 0 <= p and p in s:
1832 if 0 <= p and p in s:
1833 return False
1833 return False
1834 return True
1834 return True
1835 return subset & s.filter(filter, condrepr='<roots>')
1835 return subset & s.filter(filter, condrepr='<roots>')
1836
1836
1837 _sortkeyfuncs = {
1837 _sortkeyfuncs = {
1838 'rev': lambda c: c.rev(),
1838 'rev': lambda c: c.rev(),
1839 'branch': lambda c: c.branch(),
1839 'branch': lambda c: c.branch(),
1840 'desc': lambda c: c.description(),
1840 'desc': lambda c: c.description(),
1841 'user': lambda c: c.user(),
1841 'user': lambda c: c.user(),
1842 'author': lambda c: c.user(),
1842 'author': lambda c: c.user(),
1843 'date': lambda c: c.date()[0],
1843 'date': lambda c: c.date()[0],
1844 }
1844 }
1845
1845
1846 @predicate('sort(set[, [-]key... [, ...]])', safe=True)
1846 @predicate('sort(set[, [-]key... [, ...]])', safe=True)
1847 def sort(repo, subset, x):
1847 def sort(repo, subset, x):
1848 """Sort set by keys. The default sort order is ascending, specify a key
1848 """Sort set by keys. The default sort order is ascending, specify a key
1849 as ``-key`` to sort in descending order.
1849 as ``-key`` to sort in descending order.
1850
1850
1851 The keys can be:
1851 The keys can be:
1852
1852
1853 - ``rev`` for the revision number,
1853 - ``rev`` for the revision number,
1854 - ``branch`` for the branch name,
1854 - ``branch`` for the branch name,
1855 - ``desc`` for the commit message (description),
1855 - ``desc`` for the commit message (description),
1856 - ``user`` for user name (``author`` can be used as an alias),
1856 - ``user`` for user name (``author`` can be used as an alias),
1857 - ``date`` for the commit date
1857 - ``date`` for the commit date
1858 - ``topo`` for a reverse topographical sort
1858 - ``topo`` for a reverse topographical sort
1859
1859
1860 The ``topo`` sort order cannot be combined with other sort keys. This sort
1860 The ``topo`` sort order cannot be combined with other sort keys. This sort
1861 takes one optional argument, ``topo.firstbranch``, which takes a revset that
1861 takes one optional argument, ``topo.firstbranch``, which takes a revset that
1862 specifies what topographical branches to prioritize in the sort.
1862 specifies what topographical branches to prioritize in the sort.
1863
1863
1864 """
1864 """
1865 args = getargsdict(x, 'sort', 'set keys topo.firstbranch')
1865 args = getargsdict(x, 'sort', 'set keys topo.firstbranch')
1866 if 'set' not in args:
1866 if 'set' not in args:
1867 # i18n: "sort" is a keyword
1867 # i18n: "sort" is a keyword
1868 raise error.ParseError(_('sort requires one or two arguments'))
1868 raise error.ParseError(_('sort requires one or two arguments'))
1869 keys = "rev"
1869 keys = "rev"
1870 if 'keys' in args:
1870 if 'keys' in args:
1871 # i18n: "sort" is a keyword
1871 # i18n: "sort" is a keyword
1872 keys = getstring(args['keys'], _("sort spec must be a string"))
1872 keys = getstring(args['keys'], _("sort spec must be a string"))
1873
1873
1874 s = args['set']
1874 s = args['set']
1875 keys = keys.split()
1875 keys = keys.split()
1876 revs = getset(repo, subset, s)
1876 revs = getset(repo, subset, s)
1877
1877
1878 if len(keys) > 1 and any(k.lstrip('-') == 'topo' for k in keys):
1878 if len(keys) > 1 and any(k.lstrip('-') == 'topo' for k in keys):
1879 # i18n: "topo" is a keyword
1879 # i18n: "topo" is a keyword
1880 raise error.ParseError(_(
1880 raise error.ParseError(_(
1881 'topo sort order cannot be combined with other sort keys'))
1881 'topo sort order cannot be combined with other sort keys'))
1882
1882
1883 firstbranch = ()
1883 firstbranch = ()
1884 if 'topo.firstbranch' in args:
1884 if 'topo.firstbranch' in args:
1885 if any(k.lstrip('-') == 'topo' for k in keys):
1885 if any(k.lstrip('-') == 'topo' for k in keys):
1886 firstbranch = getset(repo, subset, args['topo.firstbranch'])
1886 firstbranch = getset(repo, subset, args['topo.firstbranch'])
1887 else:
1887 else:
1888 # i18n: "topo" and "topo.firstbranch" are keywords
1888 # i18n: "topo" and "topo.firstbranch" are keywords
1889 raise error.ParseError(_(
1889 raise error.ParseError(_(
1890 'topo.firstbranch can only be used when using the topo sort '
1890 'topo.firstbranch can only be used when using the topo sort '
1891 'key'))
1891 'key'))
1892
1892
1893 if not keys:
1894 return revs
1893 if keys == ["rev"]:
1895 if keys == ["rev"]:
1894 revs.sort()
1896 revs.sort()
1895 return revs
1897 return revs
1896 elif keys == ["-rev"]:
1898 elif keys == ["-rev"]:
1897 revs.sort(reverse=True)
1899 revs.sort(reverse=True)
1898 return revs
1900 return revs
1899 elif keys[0] in ("topo", "-topo"):
1901 elif keys[0] in ("topo", "-topo"):
1900 revs = baseset(_toposort(revs, repo.changelog.parentrevs, firstbranch),
1902 revs = baseset(_toposort(revs, repo.changelog.parentrevs, firstbranch),
1901 istopo=True)
1903 istopo=True)
1902 if keys[0][0] == '-':
1904 if keys[0][0] == '-':
1903 revs.reverse()
1905 revs.reverse()
1904 return revs
1906 return revs
1905
1907
1906 # sort() is guaranteed to be stable
1908 # sort() is guaranteed to be stable
1907 ctxs = [repo[r] for r in revs]
1909 ctxs = [repo[r] for r in revs]
1908 for k in reversed(keys):
1910 for k in reversed(keys):
1909 fk = k
1911 fk = k
1910 reverse = (k[0] == '-')
1912 reverse = (k[0] == '-')
1911 if reverse:
1913 if reverse:
1912 k = k[1:]
1914 k = k[1:]
1913 try:
1915 try:
1914 ctxs.sort(key=_sortkeyfuncs[k], reverse=reverse)
1916 ctxs.sort(key=_sortkeyfuncs[k], reverse=reverse)
1915 except KeyError:
1917 except KeyError:
1916 raise error.ParseError(_("unknown sort key %r") % fk)
1918 raise error.ParseError(_("unknown sort key %r") % fk)
1917 return baseset([c.rev() for c in ctxs])
1919 return baseset([c.rev() for c in ctxs])
1918
1920
1919 def _toposort(revs, parentsfunc, firstbranch=()):
1921 def _toposort(revs, parentsfunc, firstbranch=()):
1920 """Yield revisions from heads to roots one (topo) branch at a time.
1922 """Yield revisions from heads to roots one (topo) branch at a time.
1921
1923
1922 This function aims to be used by a graph generator that wishes to minimize
1924 This function aims to be used by a graph generator that wishes to minimize
1923 the number of parallel branches and their interleaving.
1925 the number of parallel branches and their interleaving.
1924
1926
1925 Example iteration order (numbers show the "true" order in a changelog):
1927 Example iteration order (numbers show the "true" order in a changelog):
1926
1928
1927 o 4
1929 o 4
1928 |
1930 |
1929 o 1
1931 o 1
1930 |
1932 |
1931 | o 3
1933 | o 3
1932 | |
1934 | |
1933 | o 2
1935 | o 2
1934 |/
1936 |/
1935 o 0
1937 o 0
1936
1938
1937 Note that the ancestors of merges are understood by the current
1939 Note that the ancestors of merges are understood by the current
1938 algorithm to be on the same branch. This means no reordering will
1940 algorithm to be on the same branch. This means no reordering will
1939 occur behind a merge.
1941 occur behind a merge.
1940 """
1942 """
1941
1943
1942 ### Quick summary of the algorithm
1944 ### Quick summary of the algorithm
1943 #
1945 #
1944 # This function is based around a "retention" principle. We keep revisions
1946 # This function is based around a "retention" principle. We keep revisions
1945 # in memory until we are ready to emit a whole branch that immediately
1947 # in memory until we are ready to emit a whole branch that immediately
1946 # "merges" into an existing one. This reduces the number of parallel
1948 # "merges" into an existing one. This reduces the number of parallel
1947 # branches with interleaved revisions.
1949 # branches with interleaved revisions.
1948 #
1950 #
1949 # During iteration revs are split into two groups:
1951 # During iteration revs are split into two groups:
1950 # A) revision already emitted
1952 # A) revision already emitted
1951 # B) revision in "retention". They are stored as different subgroups.
1953 # B) revision in "retention". They are stored as different subgroups.
1952 #
1954 #
1953 # for each REV, we do the following logic:
1955 # for each REV, we do the following logic:
1954 #
1956 #
1955 # 1) if REV is a parent of (A), we will emit it. If there is a
1957 # 1) if REV is a parent of (A), we will emit it. If there is a
1956 # retention group ((B) above) that is blocked on REV being
1958 # retention group ((B) above) that is blocked on REV being
1957 # available, we emit all the revisions out of that retention
1959 # available, we emit all the revisions out of that retention
1958 # group first.
1960 # group first.
1959 #
1961 #
1960 # 2) else, we'll search for a subgroup in (B) awaiting for REV to be
1962 # 2) else, we'll search for a subgroup in (B) awaiting for REV to be
1961 # available, if such subgroup exist, we add REV to it and the subgroup is
1963 # available, if such subgroup exist, we add REV to it and the subgroup is
1962 # now awaiting for REV.parents() to be available.
1964 # now awaiting for REV.parents() to be available.
1963 #
1965 #
1964 # 3) finally if no such group existed in (B), we create a new subgroup.
1966 # 3) finally if no such group existed in (B), we create a new subgroup.
1965 #
1967 #
1966 #
1968 #
1967 # To bootstrap the algorithm, we emit the tipmost revision (which
1969 # To bootstrap the algorithm, we emit the tipmost revision (which
1968 # puts it in group (A) from above).
1970 # puts it in group (A) from above).
1969
1971
1970 revs.sort(reverse=True)
1972 revs.sort(reverse=True)
1971
1973
1972 # Set of parents of revision that have been emitted. They can be considered
1974 # Set of parents of revision that have been emitted. They can be considered
1973 # unblocked as the graph generator is already aware of them so there is no
1975 # unblocked as the graph generator is already aware of them so there is no
1974 # need to delay the revisions that reference them.
1976 # need to delay the revisions that reference them.
1975 #
1977 #
1976 # If someone wants to prioritize a branch over the others, pre-filling this
1978 # If someone wants to prioritize a branch over the others, pre-filling this
1977 # set will force all other branches to wait until this branch is ready to be
1979 # set will force all other branches to wait until this branch is ready to be
1978 # emitted.
1980 # emitted.
1979 unblocked = set(firstbranch)
1981 unblocked = set(firstbranch)
1980
1982
1981 # list of groups waiting to be displayed, each group is defined by:
1983 # list of groups waiting to be displayed, each group is defined by:
1982 #
1984 #
1983 # (revs: lists of revs waiting to be displayed,
1985 # (revs: lists of revs waiting to be displayed,
1984 # blocked: set of that cannot be displayed before those in 'revs')
1986 # blocked: set of that cannot be displayed before those in 'revs')
1985 #
1987 #
1986 # The second value ('blocked') correspond to parents of any revision in the
1988 # The second value ('blocked') correspond to parents of any revision in the
1987 # group ('revs') that is not itself contained in the group. The main idea
1989 # group ('revs') that is not itself contained in the group. The main idea
1988 # of this algorithm is to delay as much as possible the emission of any
1990 # of this algorithm is to delay as much as possible the emission of any
1989 # revision. This means waiting for the moment we are about to display
1991 # revision. This means waiting for the moment we are about to display
1990 # these parents to display the revs in a group.
1992 # these parents to display the revs in a group.
1991 #
1993 #
1992 # This first implementation is smart until it encounters a merge: it will
1994 # This first implementation is smart until it encounters a merge: it will
1993 # emit revs as soon as any parent is about to be emitted and can grow an
1995 # emit revs as soon as any parent is about to be emitted and can grow an
1994 # arbitrary number of revs in 'blocked'. In practice this mean we properly
1996 # arbitrary number of revs in 'blocked'. In practice this mean we properly
1995 # retains new branches but gives up on any special ordering for ancestors
1997 # retains new branches but gives up on any special ordering for ancestors
1996 # of merges. The implementation can be improved to handle this better.
1998 # of merges. The implementation can be improved to handle this better.
1997 #
1999 #
1998 # The first subgroup is special. It corresponds to all the revision that
2000 # The first subgroup is special. It corresponds to all the revision that
1999 # were already emitted. The 'revs' lists is expected to be empty and the
2001 # were already emitted. The 'revs' lists is expected to be empty and the
2000 # 'blocked' set contains the parents revisions of already emitted revision.
2002 # 'blocked' set contains the parents revisions of already emitted revision.
2001 #
2003 #
2002 # You could pre-seed the <parents> set of groups[0] to a specific
2004 # You could pre-seed the <parents> set of groups[0] to a specific
2003 # changesets to select what the first emitted branch should be.
2005 # changesets to select what the first emitted branch should be.
2004 groups = [([], unblocked)]
2006 groups = [([], unblocked)]
2005 pendingheap = []
2007 pendingheap = []
2006 pendingset = set()
2008 pendingset = set()
2007
2009
2008 heapq.heapify(pendingheap)
2010 heapq.heapify(pendingheap)
2009 heappop = heapq.heappop
2011 heappop = heapq.heappop
2010 heappush = heapq.heappush
2012 heappush = heapq.heappush
2011 for currentrev in revs:
2013 for currentrev in revs:
2012 # Heap works with smallest element, we want highest so we invert
2014 # Heap works with smallest element, we want highest so we invert
2013 if currentrev not in pendingset:
2015 if currentrev not in pendingset:
2014 heappush(pendingheap, -currentrev)
2016 heappush(pendingheap, -currentrev)
2015 pendingset.add(currentrev)
2017 pendingset.add(currentrev)
2016 # iterates on pending rev until after the current rev have been
2018 # iterates on pending rev until after the current rev have been
2017 # processed.
2019 # processed.
2018 rev = None
2020 rev = None
2019 while rev != currentrev:
2021 while rev != currentrev:
2020 rev = -heappop(pendingheap)
2022 rev = -heappop(pendingheap)
2021 pendingset.remove(rev)
2023 pendingset.remove(rev)
2022
2024
2023 # Seek for a subgroup blocked, waiting for the current revision.
2025 # Seek for a subgroup blocked, waiting for the current revision.
2024 matching = [i for i, g in enumerate(groups) if rev in g[1]]
2026 matching = [i for i, g in enumerate(groups) if rev in g[1]]
2025
2027
2026 if matching:
2028 if matching:
2027 # The main idea is to gather together all sets that are blocked
2029 # The main idea is to gather together all sets that are blocked
2028 # on the same revision.
2030 # on the same revision.
2029 #
2031 #
2030 # Groups are merged when a common blocking ancestor is
2032 # Groups are merged when a common blocking ancestor is
2031 # observed. For example, given two groups:
2033 # observed. For example, given two groups:
2032 #
2034 #
2033 # revs [5, 4] waiting for 1
2035 # revs [5, 4] waiting for 1
2034 # revs [3, 2] waiting for 1
2036 # revs [3, 2] waiting for 1
2035 #
2037 #
2036 # These two groups will be merged when we process
2038 # These two groups will be merged when we process
2037 # 1. In theory, we could have merged the groups when
2039 # 1. In theory, we could have merged the groups when
2038 # we added 2 to the group it is now in (we could have
2040 # we added 2 to the group it is now in (we could have
2039 # noticed the groups were both blocked on 1 then), but
2041 # noticed the groups were both blocked on 1 then), but
2040 # the way it works now makes the algorithm simpler.
2042 # the way it works now makes the algorithm simpler.
2041 #
2043 #
2042 # We also always keep the oldest subgroup first. We can
2044 # We also always keep the oldest subgroup first. We can
2043 # probably improve the behavior by having the longest set
2045 # probably improve the behavior by having the longest set
2044 # first. That way, graph algorithms could minimise the length
2046 # first. That way, graph algorithms could minimise the length
2045 # of parallel lines their drawing. This is currently not done.
2047 # of parallel lines their drawing. This is currently not done.
2046 targetidx = matching.pop(0)
2048 targetidx = matching.pop(0)
2047 trevs, tparents = groups[targetidx]
2049 trevs, tparents = groups[targetidx]
2048 for i in matching:
2050 for i in matching:
2049 gr = groups[i]
2051 gr = groups[i]
2050 trevs.extend(gr[0])
2052 trevs.extend(gr[0])
2051 tparents |= gr[1]
2053 tparents |= gr[1]
2052 # delete all merged subgroups (except the one we kept)
2054 # delete all merged subgroups (except the one we kept)
2053 # (starting from the last subgroup for performance and
2055 # (starting from the last subgroup for performance and
2054 # sanity reasons)
2056 # sanity reasons)
2055 for i in reversed(matching):
2057 for i in reversed(matching):
2056 del groups[i]
2058 del groups[i]
2057 else:
2059 else:
2058 # This is a new head. We create a new subgroup for it.
2060 # This is a new head. We create a new subgroup for it.
2059 targetidx = len(groups)
2061 targetidx = len(groups)
2060 groups.append(([], set([rev])))
2062 groups.append(([], set([rev])))
2061
2063
2062 gr = groups[targetidx]
2064 gr = groups[targetidx]
2063
2065
2064 # We now add the current nodes to this subgroups. This is done
2066 # We now add the current nodes to this subgroups. This is done
2065 # after the subgroup merging because all elements from a subgroup
2067 # after the subgroup merging because all elements from a subgroup
2066 # that relied on this rev must precede it.
2068 # that relied on this rev must precede it.
2067 #
2069 #
2068 # we also update the <parents> set to include the parents of the
2070 # we also update the <parents> set to include the parents of the
2069 # new nodes.
2071 # new nodes.
2070 if rev == currentrev: # only display stuff in rev
2072 if rev == currentrev: # only display stuff in rev
2071 gr[0].append(rev)
2073 gr[0].append(rev)
2072 gr[1].remove(rev)
2074 gr[1].remove(rev)
2073 parents = [p for p in parentsfunc(rev) if p > node.nullrev]
2075 parents = [p for p in parentsfunc(rev) if p > node.nullrev]
2074 gr[1].update(parents)
2076 gr[1].update(parents)
2075 for p in parents:
2077 for p in parents:
2076 if p not in pendingset:
2078 if p not in pendingset:
2077 pendingset.add(p)
2079 pendingset.add(p)
2078 heappush(pendingheap, -p)
2080 heappush(pendingheap, -p)
2079
2081
2080 # Look for a subgroup to display
2082 # Look for a subgroup to display
2081 #
2083 #
2082 # When unblocked is empty (if clause), we were not waiting for any
2084 # When unblocked is empty (if clause), we were not waiting for any
2083 # revisions during the first iteration (if no priority was given) or
2085 # revisions during the first iteration (if no priority was given) or
2084 # if we emitted a whole disconnected set of the graph (reached a
2086 # if we emitted a whole disconnected set of the graph (reached a
2085 # root). In that case we arbitrarily take the oldest known
2087 # root). In that case we arbitrarily take the oldest known
2086 # subgroup. The heuristic could probably be better.
2088 # subgroup. The heuristic could probably be better.
2087 #
2089 #
2088 # Otherwise (elif clause) if the subgroup is blocked on
2090 # Otherwise (elif clause) if the subgroup is blocked on
2089 # a revision we just emitted, we can safely emit it as
2091 # a revision we just emitted, we can safely emit it as
2090 # well.
2092 # well.
2091 if not unblocked:
2093 if not unblocked:
2092 if len(groups) > 1: # display other subset
2094 if len(groups) > 1: # display other subset
2093 targetidx = 1
2095 targetidx = 1
2094 gr = groups[1]
2096 gr = groups[1]
2095 elif not gr[1] & unblocked:
2097 elif not gr[1] & unblocked:
2096 gr = None
2098 gr = None
2097
2099
2098 if gr is not None:
2100 if gr is not None:
2099 # update the set of awaited revisions with the one from the
2101 # update the set of awaited revisions with the one from the
2100 # subgroup
2102 # subgroup
2101 unblocked |= gr[1]
2103 unblocked |= gr[1]
2102 # output all revisions in the subgroup
2104 # output all revisions in the subgroup
2103 for r in gr[0]:
2105 for r in gr[0]:
2104 yield r
2106 yield r
2105 # delete the subgroup that you just output
2107 # delete the subgroup that you just output
2106 # unless it is groups[0] in which case you just empty it.
2108 # unless it is groups[0] in which case you just empty it.
2107 if targetidx:
2109 if targetidx:
2108 del groups[targetidx]
2110 del groups[targetidx]
2109 else:
2111 else:
2110 gr[0][:] = []
2112 gr[0][:] = []
2111 # Check if we have some subgroup waiting for revisions we are not going to
2113 # Check if we have some subgroup waiting for revisions we are not going to
2112 # iterate over
2114 # iterate over
2113 for g in groups:
2115 for g in groups:
2114 for r in g[0]:
2116 for r in g[0]:
2115 yield r
2117 yield r
2116
2118
2117 @predicate('subrepo([pattern])')
2119 @predicate('subrepo([pattern])')
2118 def subrepo(repo, subset, x):
2120 def subrepo(repo, subset, x):
2119 """Changesets that add, modify or remove the given subrepo. If no subrepo
2121 """Changesets that add, modify or remove the given subrepo. If no subrepo
2120 pattern is named, any subrepo changes are returned.
2122 pattern is named, any subrepo changes are returned.
2121 """
2123 """
2122 # i18n: "subrepo" is a keyword
2124 # i18n: "subrepo" is a keyword
2123 args = getargs(x, 0, 1, _('subrepo takes at most one argument'))
2125 args = getargs(x, 0, 1, _('subrepo takes at most one argument'))
2124 pat = None
2126 pat = None
2125 if len(args) != 0:
2127 if len(args) != 0:
2126 pat = getstring(args[0], _("subrepo requires a pattern"))
2128 pat = getstring(args[0], _("subrepo requires a pattern"))
2127
2129
2128 m = matchmod.exact(repo.root, repo.root, ['.hgsubstate'])
2130 m = matchmod.exact(repo.root, repo.root, ['.hgsubstate'])
2129
2131
2130 def submatches(names):
2132 def submatches(names):
2131 k, p, m = util.stringmatcher(pat)
2133 k, p, m = util.stringmatcher(pat)
2132 for name in names:
2134 for name in names:
2133 if m(name):
2135 if m(name):
2134 yield name
2136 yield name
2135
2137
2136 def matches(x):
2138 def matches(x):
2137 c = repo[x]
2139 c = repo[x]
2138 s = repo.status(c.p1().node(), c.node(), match=m)
2140 s = repo.status(c.p1().node(), c.node(), match=m)
2139
2141
2140 if pat is None:
2142 if pat is None:
2141 return s.added or s.modified or s.removed
2143 return s.added or s.modified or s.removed
2142
2144
2143 if s.added:
2145 if s.added:
2144 return any(submatches(c.substate.keys()))
2146 return any(submatches(c.substate.keys()))
2145
2147
2146 if s.modified:
2148 if s.modified:
2147 subs = set(c.p1().substate.keys())
2149 subs = set(c.p1().substate.keys())
2148 subs.update(c.substate.keys())
2150 subs.update(c.substate.keys())
2149
2151
2150 for path in submatches(subs):
2152 for path in submatches(subs):
2151 if c.p1().substate.get(path) != c.substate.get(path):
2153 if c.p1().substate.get(path) != c.substate.get(path):
2152 return True
2154 return True
2153
2155
2154 if s.removed:
2156 if s.removed:
2155 return any(submatches(c.p1().substate.keys()))
2157 return any(submatches(c.p1().substate.keys()))
2156
2158
2157 return False
2159 return False
2158
2160
2159 return subset.filter(matches, condrepr=('<subrepo %r>', pat))
2161 return subset.filter(matches, condrepr=('<subrepo %r>', pat))
2160
2162
2161 def _substringmatcher(pattern):
2163 def _substringmatcher(pattern):
2162 kind, pattern, matcher = util.stringmatcher(pattern)
2164 kind, pattern, matcher = util.stringmatcher(pattern)
2163 if kind == 'literal':
2165 if kind == 'literal':
2164 matcher = lambda s: pattern in s
2166 matcher = lambda s: pattern in s
2165 return kind, pattern, matcher
2167 return kind, pattern, matcher
2166
2168
2167 @predicate('tag([name])', safe=True)
2169 @predicate('tag([name])', safe=True)
2168 def tag(repo, subset, x):
2170 def tag(repo, subset, x):
2169 """The specified tag by name, or all tagged revisions if no name is given.
2171 """The specified tag by name, or all tagged revisions if no name is given.
2170
2172
2171 If `name` starts with `re:`, the remainder of the name is treated as
2173 If `name` starts with `re:`, the remainder of the name is treated as
2172 a regular expression. To match a tag that actually starts with `re:`,
2174 a regular expression. To match a tag that actually starts with `re:`,
2173 use the prefix `literal:`.
2175 use the prefix `literal:`.
2174 """
2176 """
2175 # i18n: "tag" is a keyword
2177 # i18n: "tag" is a keyword
2176 args = getargs(x, 0, 1, _("tag takes one or no arguments"))
2178 args = getargs(x, 0, 1, _("tag takes one or no arguments"))
2177 cl = repo.changelog
2179 cl = repo.changelog
2178 if args:
2180 if args:
2179 pattern = getstring(args[0],
2181 pattern = getstring(args[0],
2180 # i18n: "tag" is a keyword
2182 # i18n: "tag" is a keyword
2181 _('the argument to tag must be a string'))
2183 _('the argument to tag must be a string'))
2182 kind, pattern, matcher = util.stringmatcher(pattern)
2184 kind, pattern, matcher = util.stringmatcher(pattern)
2183 if kind == 'literal':
2185 if kind == 'literal':
2184 # avoid resolving all tags
2186 # avoid resolving all tags
2185 tn = repo._tagscache.tags.get(pattern, None)
2187 tn = repo._tagscache.tags.get(pattern, None)
2186 if tn is None:
2188 if tn is None:
2187 raise error.RepoLookupError(_("tag '%s' does not exist")
2189 raise error.RepoLookupError(_("tag '%s' does not exist")
2188 % pattern)
2190 % pattern)
2189 s = set([repo[tn].rev()])
2191 s = set([repo[tn].rev()])
2190 else:
2192 else:
2191 s = set([cl.rev(n) for t, n in repo.tagslist() if matcher(t)])
2193 s = set([cl.rev(n) for t, n in repo.tagslist() if matcher(t)])
2192 else:
2194 else:
2193 s = set([cl.rev(n) for t, n in repo.tagslist() if t != 'tip'])
2195 s = set([cl.rev(n) for t, n in repo.tagslist() if t != 'tip'])
2194 return subset & s
2196 return subset & s
2195
2197
2196 @predicate('tagged', safe=True)
2198 @predicate('tagged', safe=True)
2197 def tagged(repo, subset, x):
2199 def tagged(repo, subset, x):
2198 return tag(repo, subset, x)
2200 return tag(repo, subset, x)
2199
2201
2200 @predicate('unstable()', safe=True)
2202 @predicate('unstable()', safe=True)
2201 def unstable(repo, subset, x):
2203 def unstable(repo, subset, x):
2202 """Non-obsolete changesets with obsolete ancestors.
2204 """Non-obsolete changesets with obsolete ancestors.
2203 """
2205 """
2204 # i18n: "unstable" is a keyword
2206 # i18n: "unstable" is a keyword
2205 getargs(x, 0, 0, _("unstable takes no arguments"))
2207 getargs(x, 0, 0, _("unstable takes no arguments"))
2206 unstables = obsmod.getrevs(repo, 'unstable')
2208 unstables = obsmod.getrevs(repo, 'unstable')
2207 return subset & unstables
2209 return subset & unstables
2208
2210
2209
2211
2210 @predicate('user(string)', safe=True)
2212 @predicate('user(string)', safe=True)
2211 def user(repo, subset, x):
2213 def user(repo, subset, x):
2212 """User name contains string. The match is case-insensitive.
2214 """User name contains string. The match is case-insensitive.
2213
2215
2214 If `string` starts with `re:`, the remainder of the string is treated as
2216 If `string` starts with `re:`, the remainder of the string is treated as
2215 a regular expression. To match a user that actually contains `re:`, use
2217 a regular expression. To match a user that actually contains `re:`, use
2216 the prefix `literal:`.
2218 the prefix `literal:`.
2217 """
2219 """
2218 return author(repo, subset, x)
2220 return author(repo, subset, x)
2219
2221
2220 # experimental
2222 # experimental
2221 @predicate('wdir', safe=True)
2223 @predicate('wdir', safe=True)
2222 def wdir(repo, subset, x):
2224 def wdir(repo, subset, x):
2223 # i18n: "wdir" is a keyword
2225 # i18n: "wdir" is a keyword
2224 getargs(x, 0, 0, _("wdir takes no arguments"))
2226 getargs(x, 0, 0, _("wdir takes no arguments"))
2225 if node.wdirrev in subset or isinstance(subset, fullreposet):
2227 if node.wdirrev in subset or isinstance(subset, fullreposet):
2226 return baseset([node.wdirrev])
2228 return baseset([node.wdirrev])
2227 return baseset()
2229 return baseset()
2228
2230
2229 # for internal use
2231 # for internal use
2230 @predicate('_list', safe=True)
2232 @predicate('_list', safe=True)
2231 def _list(repo, subset, x):
2233 def _list(repo, subset, x):
2232 s = getstring(x, "internal error")
2234 s = getstring(x, "internal error")
2233 if not s:
2235 if not s:
2234 return baseset()
2236 return baseset()
2235 # remove duplicates here. it's difficult for caller to deduplicate sets
2237 # remove duplicates here. it's difficult for caller to deduplicate sets
2236 # because different symbols can point to the same rev.
2238 # because different symbols can point to the same rev.
2237 cl = repo.changelog
2239 cl = repo.changelog
2238 ls = []
2240 ls = []
2239 seen = set()
2241 seen = set()
2240 for t in s.split('\0'):
2242 for t in s.split('\0'):
2241 try:
2243 try:
2242 # fast path for integer revision
2244 # fast path for integer revision
2243 r = int(t)
2245 r = int(t)
2244 if str(r) != t or r not in cl:
2246 if str(r) != t or r not in cl:
2245 raise ValueError
2247 raise ValueError
2246 revs = [r]
2248 revs = [r]
2247 except ValueError:
2249 except ValueError:
2248 revs = stringset(repo, subset, t)
2250 revs = stringset(repo, subset, t)
2249
2251
2250 for r in revs:
2252 for r in revs:
2251 if r in seen:
2253 if r in seen:
2252 continue
2254 continue
2253 if (r in subset
2255 if (r in subset
2254 or r == node.nullrev and isinstance(subset, fullreposet)):
2256 or r == node.nullrev and isinstance(subset, fullreposet)):
2255 ls.append(r)
2257 ls.append(r)
2256 seen.add(r)
2258 seen.add(r)
2257 return baseset(ls)
2259 return baseset(ls)
2258
2260
2259 # for internal use
2261 # for internal use
2260 @predicate('_intlist', safe=True)
2262 @predicate('_intlist', safe=True)
2261 def _intlist(repo, subset, x):
2263 def _intlist(repo, subset, x):
2262 s = getstring(x, "internal error")
2264 s = getstring(x, "internal error")
2263 if not s:
2265 if not s:
2264 return baseset()
2266 return baseset()
2265 ls = [int(r) for r in s.split('\0')]
2267 ls = [int(r) for r in s.split('\0')]
2266 s = subset
2268 s = subset
2267 return baseset([r for r in ls if r in s])
2269 return baseset([r for r in ls if r in s])
2268
2270
2269 # for internal use
2271 # for internal use
2270 @predicate('_hexlist', safe=True)
2272 @predicate('_hexlist', safe=True)
2271 def _hexlist(repo, subset, x):
2273 def _hexlist(repo, subset, x):
2272 s = getstring(x, "internal error")
2274 s = getstring(x, "internal error")
2273 if not s:
2275 if not s:
2274 return baseset()
2276 return baseset()
2275 cl = repo.changelog
2277 cl = repo.changelog
2276 ls = [cl.rev(node.bin(r)) for r in s.split('\0')]
2278 ls = [cl.rev(node.bin(r)) for r in s.split('\0')]
2277 s = subset
2279 s = subset
2278 return baseset([r for r in ls if r in s])
2280 return baseset([r for r in ls if r in s])
2279
2281
2280 methods = {
2282 methods = {
2281 "range": rangeset,
2283 "range": rangeset,
2282 "dagrange": dagrange,
2284 "dagrange": dagrange,
2283 "string": stringset,
2285 "string": stringset,
2284 "symbol": stringset,
2286 "symbol": stringset,
2285 "and": andset,
2287 "and": andset,
2286 "or": orset,
2288 "or": orset,
2287 "not": notset,
2289 "not": notset,
2288 "difference": differenceset,
2290 "difference": differenceset,
2289 "list": listset,
2291 "list": listset,
2290 "keyvalue": keyvaluepair,
2292 "keyvalue": keyvaluepair,
2291 "func": func,
2293 "func": func,
2292 "ancestor": ancestorspec,
2294 "ancestor": ancestorspec,
2293 "parent": parentspec,
2295 "parent": parentspec,
2294 "parentpost": p1,
2296 "parentpost": p1,
2295 }
2297 }
2296
2298
2297 def _matchonly(revs, bases):
2299 def _matchonly(revs, bases):
2298 """
2300 """
2299 >>> f = lambda *args: _matchonly(*map(parse, args))
2301 >>> f = lambda *args: _matchonly(*map(parse, args))
2300 >>> f('ancestors(A)', 'not ancestors(B)')
2302 >>> f('ancestors(A)', 'not ancestors(B)')
2301 ('list', ('symbol', 'A'), ('symbol', 'B'))
2303 ('list', ('symbol', 'A'), ('symbol', 'B'))
2302 """
2304 """
2303 if (revs is not None
2305 if (revs is not None
2304 and revs[0] == 'func'
2306 and revs[0] == 'func'
2305 and getstring(revs[1], _('not a symbol')) == 'ancestors'
2307 and getstring(revs[1], _('not a symbol')) == 'ancestors'
2306 and bases is not None
2308 and bases is not None
2307 and bases[0] == 'not'
2309 and bases[0] == 'not'
2308 and bases[1][0] == 'func'
2310 and bases[1][0] == 'func'
2309 and getstring(bases[1][1], _('not a symbol')) == 'ancestors'):
2311 and getstring(bases[1][1], _('not a symbol')) == 'ancestors'):
2310 return ('list', revs[2], bases[1][2])
2312 return ('list', revs[2], bases[1][2])
2311
2313
2312 def _optimize(x, small):
2314 def _optimize(x, small):
2313 if x is None:
2315 if x is None:
2314 return 0, x
2316 return 0, x
2315
2317
2316 smallbonus = 1
2318 smallbonus = 1
2317 if small:
2319 if small:
2318 smallbonus = .5
2320 smallbonus = .5
2319
2321
2320 op = x[0]
2322 op = x[0]
2321 if op == 'minus':
2323 if op == 'minus':
2322 return _optimize(('and', x[1], ('not', x[2])), small)
2324 return _optimize(('and', x[1], ('not', x[2])), small)
2323 elif op == 'only':
2325 elif op == 'only':
2324 t = ('func', ('symbol', 'only'), ('list', x[1], x[2]))
2326 t = ('func', ('symbol', 'only'), ('list', x[1], x[2]))
2325 return _optimize(t, small)
2327 return _optimize(t, small)
2326 elif op == 'onlypost':
2328 elif op == 'onlypost':
2327 return _optimize(('func', ('symbol', 'only'), x[1]), small)
2329 return _optimize(('func', ('symbol', 'only'), x[1]), small)
2328 elif op == 'dagrangepre':
2330 elif op == 'dagrangepre':
2329 return _optimize(('func', ('symbol', 'ancestors'), x[1]), small)
2331 return _optimize(('func', ('symbol', 'ancestors'), x[1]), small)
2330 elif op == 'dagrangepost':
2332 elif op == 'dagrangepost':
2331 return _optimize(('func', ('symbol', 'descendants'), x[1]), small)
2333 return _optimize(('func', ('symbol', 'descendants'), x[1]), small)
2332 elif op == 'rangeall':
2334 elif op == 'rangeall':
2333 return _optimize(('range', ('string', '0'), ('string', 'tip')), small)
2335 return _optimize(('range', ('string', '0'), ('string', 'tip')), small)
2334 elif op == 'rangepre':
2336 elif op == 'rangepre':
2335 return _optimize(('range', ('string', '0'), x[1]), small)
2337 return _optimize(('range', ('string', '0'), x[1]), small)
2336 elif op == 'rangepost':
2338 elif op == 'rangepost':
2337 return _optimize(('range', x[1], ('string', 'tip')), small)
2339 return _optimize(('range', x[1], ('string', 'tip')), small)
2338 elif op == 'negate':
2340 elif op == 'negate':
2339 s = getstring(x[1], _("can't negate that"))
2341 s = getstring(x[1], _("can't negate that"))
2340 return _optimize(('string', '-' + s), small)
2342 return _optimize(('string', '-' + s), small)
2341 elif op in 'string symbol negate':
2343 elif op in 'string symbol negate':
2342 return smallbonus, x # single revisions are small
2344 return smallbonus, x # single revisions are small
2343 elif op == 'and':
2345 elif op == 'and':
2344 wa, ta = _optimize(x[1], True)
2346 wa, ta = _optimize(x[1], True)
2345 wb, tb = _optimize(x[2], True)
2347 wb, tb = _optimize(x[2], True)
2346 w = min(wa, wb)
2348 w = min(wa, wb)
2347
2349
2348 # (::x and not ::y)/(not ::y and ::x) have a fast path
2350 # (::x and not ::y)/(not ::y and ::x) have a fast path
2349 tm = _matchonly(ta, tb) or _matchonly(tb, ta)
2351 tm = _matchonly(ta, tb) or _matchonly(tb, ta)
2350 if tm:
2352 if tm:
2351 return w, ('func', ('symbol', 'only'), tm)
2353 return w, ('func', ('symbol', 'only'), tm)
2352
2354
2353 if tb is not None and tb[0] == 'not':
2355 if tb is not None and tb[0] == 'not':
2354 return wa, ('difference', ta, tb[1])
2356 return wa, ('difference', ta, tb[1])
2355
2357
2356 if wa > wb:
2358 if wa > wb:
2357 return w, (op, tb, ta)
2359 return w, (op, tb, ta)
2358 return w, (op, ta, tb)
2360 return w, (op, ta, tb)
2359 elif op == 'or':
2361 elif op == 'or':
2360 # fast path for machine-generated expression, that is likely to have
2362 # fast path for machine-generated expression, that is likely to have
2361 # lots of trivial revisions: 'a + b + c()' to '_list(a b) + c()'
2363 # lots of trivial revisions: 'a + b + c()' to '_list(a b) + c()'
2362 ws, ts, ss = [], [], []
2364 ws, ts, ss = [], [], []
2363 def flushss():
2365 def flushss():
2364 if not ss:
2366 if not ss:
2365 return
2367 return
2366 if len(ss) == 1:
2368 if len(ss) == 1:
2367 w, t = ss[0]
2369 w, t = ss[0]
2368 else:
2370 else:
2369 s = '\0'.join(t[1] for w, t in ss)
2371 s = '\0'.join(t[1] for w, t in ss)
2370 y = ('func', ('symbol', '_list'), ('string', s))
2372 y = ('func', ('symbol', '_list'), ('string', s))
2371 w, t = _optimize(y, False)
2373 w, t = _optimize(y, False)
2372 ws.append(w)
2374 ws.append(w)
2373 ts.append(t)
2375 ts.append(t)
2374 del ss[:]
2376 del ss[:]
2375 for y in x[1:]:
2377 for y in x[1:]:
2376 w, t = _optimize(y, False)
2378 w, t = _optimize(y, False)
2377 if t is not None and (t[0] == 'string' or t[0] == 'symbol'):
2379 if t is not None and (t[0] == 'string' or t[0] == 'symbol'):
2378 ss.append((w, t))
2380 ss.append((w, t))
2379 continue
2381 continue
2380 flushss()
2382 flushss()
2381 ws.append(w)
2383 ws.append(w)
2382 ts.append(t)
2384 ts.append(t)
2383 flushss()
2385 flushss()
2384 if len(ts) == 1:
2386 if len(ts) == 1:
2385 return ws[0], ts[0] # 'or' operation is fully optimized out
2387 return ws[0], ts[0] # 'or' operation is fully optimized out
2386 # we can't reorder trees by weight because it would change the order.
2388 # we can't reorder trees by weight because it would change the order.
2387 # ("sort(a + b)" == "sort(b + a)", but "a + b" != "b + a")
2389 # ("sort(a + b)" == "sort(b + a)", but "a + b" != "b + a")
2388 # ts = tuple(t for w, t in sorted(zip(ws, ts), key=lambda wt: wt[0]))
2390 # ts = tuple(t for w, t in sorted(zip(ws, ts), key=lambda wt: wt[0]))
2389 return max(ws), (op,) + tuple(ts)
2391 return max(ws), (op,) + tuple(ts)
2390 elif op == 'not':
2392 elif op == 'not':
2391 # Optimize not public() to _notpublic() because we have a fast version
2393 # Optimize not public() to _notpublic() because we have a fast version
2392 if x[1] == ('func', ('symbol', 'public'), None):
2394 if x[1] == ('func', ('symbol', 'public'), None):
2393 newsym = ('func', ('symbol', '_notpublic'), None)
2395 newsym = ('func', ('symbol', '_notpublic'), None)
2394 o = _optimize(newsym, not small)
2396 o = _optimize(newsym, not small)
2395 return o[0], o[1]
2397 return o[0], o[1]
2396 else:
2398 else:
2397 o = _optimize(x[1], not small)
2399 o = _optimize(x[1], not small)
2398 return o[0], (op, o[1])
2400 return o[0], (op, o[1])
2399 elif op == 'parentpost':
2401 elif op == 'parentpost':
2400 o = _optimize(x[1], small)
2402 o = _optimize(x[1], small)
2401 return o[0], (op, o[1])
2403 return o[0], (op, o[1])
2402 elif op == 'group':
2404 elif op == 'group':
2403 return _optimize(x[1], small)
2405 return _optimize(x[1], small)
2404 elif op in 'dagrange range parent ancestorspec':
2406 elif op in 'dagrange range parent ancestorspec':
2405 if op == 'parent':
2407 if op == 'parent':
2406 # x^:y means (x^) : y, not x ^ (:y)
2408 # x^:y means (x^) : y, not x ^ (:y)
2407 post = ('parentpost', x[1])
2409 post = ('parentpost', x[1])
2408 if x[2][0] == 'dagrangepre':
2410 if x[2][0] == 'dagrangepre':
2409 return _optimize(('dagrange', post, x[2][1]), small)
2411 return _optimize(('dagrange', post, x[2][1]), small)
2410 elif x[2][0] == 'rangepre':
2412 elif x[2][0] == 'rangepre':
2411 return _optimize(('range', post, x[2][1]), small)
2413 return _optimize(('range', post, x[2][1]), small)
2412
2414
2413 wa, ta = _optimize(x[1], small)
2415 wa, ta = _optimize(x[1], small)
2414 wb, tb = _optimize(x[2], small)
2416 wb, tb = _optimize(x[2], small)
2415 return wa + wb, (op, ta, tb)
2417 return wa + wb, (op, ta, tb)
2416 elif op == 'list':
2418 elif op == 'list':
2417 ws, ts = zip(*(_optimize(y, small) for y in x[1:]))
2419 ws, ts = zip(*(_optimize(y, small) for y in x[1:]))
2418 return sum(ws), (op,) + ts
2420 return sum(ws), (op,) + ts
2419 elif op == 'func':
2421 elif op == 'func':
2420 f = getstring(x[1], _("not a symbol"))
2422 f = getstring(x[1], _("not a symbol"))
2421 wa, ta = _optimize(x[2], small)
2423 wa, ta = _optimize(x[2], small)
2422 if f in ("author branch closed date desc file grep keyword "
2424 if f in ("author branch closed date desc file grep keyword "
2423 "outgoing user"):
2425 "outgoing user"):
2424 w = 10 # slow
2426 w = 10 # slow
2425 elif f in "modifies adds removes":
2427 elif f in "modifies adds removes":
2426 w = 30 # slower
2428 w = 30 # slower
2427 elif f == "contains":
2429 elif f == "contains":
2428 w = 100 # very slow
2430 w = 100 # very slow
2429 elif f == "ancestor":
2431 elif f == "ancestor":
2430 w = 1 * smallbonus
2432 w = 1 * smallbonus
2431 elif f in "reverse limit first _intlist":
2433 elif f in "reverse limit first _intlist":
2432 w = 0
2434 w = 0
2433 elif f in "sort":
2435 elif f in "sort":
2434 w = 10 # assume most sorts look at changelog
2436 w = 10 # assume most sorts look at changelog
2435 else:
2437 else:
2436 w = 1
2438 w = 1
2437 return w + wa, (op, x[1], ta)
2439 return w + wa, (op, x[1], ta)
2438 return 1, x
2440 return 1, x
2439
2441
2440 def optimize(tree):
2442 def optimize(tree):
2441 _weight, newtree = _optimize(tree, small=True)
2443 _weight, newtree = _optimize(tree, small=True)
2442 return newtree
2444 return newtree
2443
2445
2444 # the set of valid characters for the initial letter of symbols in
2446 # the set of valid characters for the initial letter of symbols in
2445 # alias declarations and definitions
2447 # alias declarations and definitions
2446 _aliassyminitletters = set(c for c in [chr(i) for i in xrange(256)]
2448 _aliassyminitletters = set(c for c in [chr(i) for i in xrange(256)]
2447 if c.isalnum() or c in '._@$' or ord(c) > 127)
2449 if c.isalnum() or c in '._@$' or ord(c) > 127)
2448
2450
2449 def _parsewith(spec, lookup=None, syminitletters=None):
2451 def _parsewith(spec, lookup=None, syminitletters=None):
2450 """Generate a parse tree of given spec with given tokenizing options
2452 """Generate a parse tree of given spec with given tokenizing options
2451
2453
2452 >>> _parsewith('foo($1)', syminitletters=_aliassyminitletters)
2454 >>> _parsewith('foo($1)', syminitletters=_aliassyminitletters)
2453 ('func', ('symbol', 'foo'), ('symbol', '$1'))
2455 ('func', ('symbol', 'foo'), ('symbol', '$1'))
2454 >>> _parsewith('$1')
2456 >>> _parsewith('$1')
2455 Traceback (most recent call last):
2457 Traceback (most recent call last):
2456 ...
2458 ...
2457 ParseError: ("syntax error in revset '$1'", 0)
2459 ParseError: ("syntax error in revset '$1'", 0)
2458 >>> _parsewith('foo bar')
2460 >>> _parsewith('foo bar')
2459 Traceback (most recent call last):
2461 Traceback (most recent call last):
2460 ...
2462 ...
2461 ParseError: ('invalid token', 4)
2463 ParseError: ('invalid token', 4)
2462 """
2464 """
2463 p = parser.parser(elements)
2465 p = parser.parser(elements)
2464 tree, pos = p.parse(tokenize(spec, lookup=lookup,
2466 tree, pos = p.parse(tokenize(spec, lookup=lookup,
2465 syminitletters=syminitletters))
2467 syminitletters=syminitletters))
2466 if pos != len(spec):
2468 if pos != len(spec):
2467 raise error.ParseError(_('invalid token'), pos)
2469 raise error.ParseError(_('invalid token'), pos)
2468 return parser.simplifyinfixops(tree, ('list', 'or'))
2470 return parser.simplifyinfixops(tree, ('list', 'or'))
2469
2471
2470 class _aliasrules(parser.basealiasrules):
2472 class _aliasrules(parser.basealiasrules):
2471 """Parsing and expansion rule set of revset aliases"""
2473 """Parsing and expansion rule set of revset aliases"""
2472 _section = _('revset alias')
2474 _section = _('revset alias')
2473
2475
2474 @staticmethod
2476 @staticmethod
2475 def _parse(spec):
2477 def _parse(spec):
2476 """Parse alias declaration/definition ``spec``
2478 """Parse alias declaration/definition ``spec``
2477
2479
2478 This allows symbol names to use also ``$`` as an initial letter
2480 This allows symbol names to use also ``$`` as an initial letter
2479 (for backward compatibility), and callers of this function should
2481 (for backward compatibility), and callers of this function should
2480 examine whether ``$`` is used also for unexpected symbols or not.
2482 examine whether ``$`` is used also for unexpected symbols or not.
2481 """
2483 """
2482 return _parsewith(spec, syminitletters=_aliassyminitletters)
2484 return _parsewith(spec, syminitletters=_aliassyminitletters)
2483
2485
2484 @staticmethod
2486 @staticmethod
2485 def _trygetfunc(tree):
2487 def _trygetfunc(tree):
2486 if tree[0] == 'func' and tree[1][0] == 'symbol':
2488 if tree[0] == 'func' and tree[1][0] == 'symbol':
2487 return tree[1][1], getlist(tree[2])
2489 return tree[1][1], getlist(tree[2])
2488
2490
2489 def expandaliases(ui, tree, showwarning=None):
2491 def expandaliases(ui, tree, showwarning=None):
2490 aliases = _aliasrules.buildmap(ui.configitems('revsetalias'))
2492 aliases = _aliasrules.buildmap(ui.configitems('revsetalias'))
2491 tree = _aliasrules.expand(aliases, tree)
2493 tree = _aliasrules.expand(aliases, tree)
2492 if showwarning:
2494 if showwarning:
2493 # warn about problematic (but not referred) aliases
2495 # warn about problematic (but not referred) aliases
2494 for name, alias in sorted(aliases.iteritems()):
2496 for name, alias in sorted(aliases.iteritems()):
2495 if alias.error and not alias.warned:
2497 if alias.error and not alias.warned:
2496 showwarning(_('warning: %s\n') % (alias.error))
2498 showwarning(_('warning: %s\n') % (alias.error))
2497 alias.warned = True
2499 alias.warned = True
2498 return tree
2500 return tree
2499
2501
2500 def foldconcat(tree):
2502 def foldconcat(tree):
2501 """Fold elements to be concatenated by `##`
2503 """Fold elements to be concatenated by `##`
2502 """
2504 """
2503 if not isinstance(tree, tuple) or tree[0] in ('string', 'symbol'):
2505 if not isinstance(tree, tuple) or tree[0] in ('string', 'symbol'):
2504 return tree
2506 return tree
2505 if tree[0] == '_concat':
2507 if tree[0] == '_concat':
2506 pending = [tree]
2508 pending = [tree]
2507 l = []
2509 l = []
2508 while pending:
2510 while pending:
2509 e = pending.pop()
2511 e = pending.pop()
2510 if e[0] == '_concat':
2512 if e[0] == '_concat':
2511 pending.extend(reversed(e[1:]))
2513 pending.extend(reversed(e[1:]))
2512 elif e[0] in ('string', 'symbol'):
2514 elif e[0] in ('string', 'symbol'):
2513 l.append(e[1])
2515 l.append(e[1])
2514 else:
2516 else:
2515 msg = _("\"##\" can't concatenate \"%s\" element") % (e[0])
2517 msg = _("\"##\" can't concatenate \"%s\" element") % (e[0])
2516 raise error.ParseError(msg)
2518 raise error.ParseError(msg)
2517 return ('string', ''.join(l))
2519 return ('string', ''.join(l))
2518 else:
2520 else:
2519 return tuple(foldconcat(t) for t in tree)
2521 return tuple(foldconcat(t) for t in tree)
2520
2522
2521 def parse(spec, lookup=None):
2523 def parse(spec, lookup=None):
2522 return _parsewith(spec, lookup=lookup)
2524 return _parsewith(spec, lookup=lookup)
2523
2525
2524 def posttreebuilthook(tree, repo):
2526 def posttreebuilthook(tree, repo):
2525 # hook for extensions to execute code on the optimized tree
2527 # hook for extensions to execute code on the optimized tree
2526 pass
2528 pass
2527
2529
2528 def match(ui, spec, repo=None):
2530 def match(ui, spec, repo=None):
2529 if not spec:
2531 if not spec:
2530 raise error.ParseError(_("empty query"))
2532 raise error.ParseError(_("empty query"))
2531 lookup = None
2533 lookup = None
2532 if repo:
2534 if repo:
2533 lookup = repo.__contains__
2535 lookup = repo.__contains__
2534 tree = parse(spec, lookup)
2536 tree = parse(spec, lookup)
2535 return _makematcher(ui, tree, repo)
2537 return _makematcher(ui, tree, repo)
2536
2538
2537 def matchany(ui, specs, repo=None):
2539 def matchany(ui, specs, repo=None):
2538 """Create a matcher that will include any revisions matching one of the
2540 """Create a matcher that will include any revisions matching one of the
2539 given specs"""
2541 given specs"""
2540 if not specs:
2542 if not specs:
2541 def mfunc(repo, subset=None):
2543 def mfunc(repo, subset=None):
2542 return baseset()
2544 return baseset()
2543 return mfunc
2545 return mfunc
2544 if not all(specs):
2546 if not all(specs):
2545 raise error.ParseError(_("empty query"))
2547 raise error.ParseError(_("empty query"))
2546 lookup = None
2548 lookup = None
2547 if repo:
2549 if repo:
2548 lookup = repo.__contains__
2550 lookup = repo.__contains__
2549 if len(specs) == 1:
2551 if len(specs) == 1:
2550 tree = parse(specs[0], lookup)
2552 tree = parse(specs[0], lookup)
2551 else:
2553 else:
2552 tree = ('or',) + tuple(parse(s, lookup) for s in specs)
2554 tree = ('or',) + tuple(parse(s, lookup) for s in specs)
2553 return _makematcher(ui, tree, repo)
2555 return _makematcher(ui, tree, repo)
2554
2556
2555 def _makematcher(ui, tree, repo):
2557 def _makematcher(ui, tree, repo):
2556 if ui:
2558 if ui:
2557 tree = expandaliases(ui, tree, showwarning=ui.warn)
2559 tree = expandaliases(ui, tree, showwarning=ui.warn)
2558 tree = foldconcat(tree)
2560 tree = foldconcat(tree)
2559 tree = optimize(tree)
2561 tree = optimize(tree)
2560 posttreebuilthook(tree, repo)
2562 posttreebuilthook(tree, repo)
2561 def mfunc(repo, subset=None):
2563 def mfunc(repo, subset=None):
2562 if subset is None:
2564 if subset is None:
2563 subset = fullreposet(repo)
2565 subset = fullreposet(repo)
2564 if util.safehasattr(subset, 'isascending'):
2566 if util.safehasattr(subset, 'isascending'):
2565 result = getset(repo, subset, tree)
2567 result = getset(repo, subset, tree)
2566 else:
2568 else:
2567 result = getset(repo, baseset(subset), tree)
2569 result = getset(repo, baseset(subset), tree)
2568 return result
2570 return result
2569 return mfunc
2571 return mfunc
2570
2572
2571 def formatspec(expr, *args):
2573 def formatspec(expr, *args):
2572 '''
2574 '''
2573 This is a convenience function for using revsets internally, and
2575 This is a convenience function for using revsets internally, and
2574 escapes arguments appropriately. Aliases are intentionally ignored
2576 escapes arguments appropriately. Aliases are intentionally ignored
2575 so that intended expression behavior isn't accidentally subverted.
2577 so that intended expression behavior isn't accidentally subverted.
2576
2578
2577 Supported arguments:
2579 Supported arguments:
2578
2580
2579 %r = revset expression, parenthesized
2581 %r = revset expression, parenthesized
2580 %d = int(arg), no quoting
2582 %d = int(arg), no quoting
2581 %s = string(arg), escaped and single-quoted
2583 %s = string(arg), escaped and single-quoted
2582 %b = arg.branch(), escaped and single-quoted
2584 %b = arg.branch(), escaped and single-quoted
2583 %n = hex(arg), single-quoted
2585 %n = hex(arg), single-quoted
2584 %% = a literal '%'
2586 %% = a literal '%'
2585
2587
2586 Prefixing the type with 'l' specifies a parenthesized list of that type.
2588 Prefixing the type with 'l' specifies a parenthesized list of that type.
2587
2589
2588 >>> formatspec('%r:: and %lr', '10 or 11', ("this()", "that()"))
2590 >>> formatspec('%r:: and %lr', '10 or 11', ("this()", "that()"))
2589 '(10 or 11):: and ((this()) or (that()))'
2591 '(10 or 11):: and ((this()) or (that()))'
2590 >>> formatspec('%d:: and not %d::', 10, 20)
2592 >>> formatspec('%d:: and not %d::', 10, 20)
2591 '10:: and not 20::'
2593 '10:: and not 20::'
2592 >>> formatspec('%ld or %ld', [], [1])
2594 >>> formatspec('%ld or %ld', [], [1])
2593 "_list('') or 1"
2595 "_list('') or 1"
2594 >>> formatspec('keyword(%s)', 'foo\\xe9')
2596 >>> formatspec('keyword(%s)', 'foo\\xe9')
2595 "keyword('foo\\\\xe9')"
2597 "keyword('foo\\\\xe9')"
2596 >>> b = lambda: 'default'
2598 >>> b = lambda: 'default'
2597 >>> b.branch = b
2599 >>> b.branch = b
2598 >>> formatspec('branch(%b)', b)
2600 >>> formatspec('branch(%b)', b)
2599 "branch('default')"
2601 "branch('default')"
2600 >>> formatspec('root(%ls)', ['a', 'b', 'c', 'd'])
2602 >>> formatspec('root(%ls)', ['a', 'b', 'c', 'd'])
2601 "root(_list('a\\x00b\\x00c\\x00d'))"
2603 "root(_list('a\\x00b\\x00c\\x00d'))"
2602 '''
2604 '''
2603
2605
2604 def quote(s):
2606 def quote(s):
2605 return repr(str(s))
2607 return repr(str(s))
2606
2608
2607 def argtype(c, arg):
2609 def argtype(c, arg):
2608 if c == 'd':
2610 if c == 'd':
2609 return str(int(arg))
2611 return str(int(arg))
2610 elif c == 's':
2612 elif c == 's':
2611 return quote(arg)
2613 return quote(arg)
2612 elif c == 'r':
2614 elif c == 'r':
2613 parse(arg) # make sure syntax errors are confined
2615 parse(arg) # make sure syntax errors are confined
2614 return '(%s)' % arg
2616 return '(%s)' % arg
2615 elif c == 'n':
2617 elif c == 'n':
2616 return quote(node.hex(arg))
2618 return quote(node.hex(arg))
2617 elif c == 'b':
2619 elif c == 'b':
2618 return quote(arg.branch())
2620 return quote(arg.branch())
2619
2621
2620 def listexp(s, t):
2622 def listexp(s, t):
2621 l = len(s)
2623 l = len(s)
2622 if l == 0:
2624 if l == 0:
2623 return "_list('')"
2625 return "_list('')"
2624 elif l == 1:
2626 elif l == 1:
2625 return argtype(t, s[0])
2627 return argtype(t, s[0])
2626 elif t == 'd':
2628 elif t == 'd':
2627 return "_intlist('%s')" % "\0".join(str(int(a)) for a in s)
2629 return "_intlist('%s')" % "\0".join(str(int(a)) for a in s)
2628 elif t == 's':
2630 elif t == 's':
2629 return "_list('%s')" % "\0".join(s)
2631 return "_list('%s')" % "\0".join(s)
2630 elif t == 'n':
2632 elif t == 'n':
2631 return "_hexlist('%s')" % "\0".join(node.hex(a) for a in s)
2633 return "_hexlist('%s')" % "\0".join(node.hex(a) for a in s)
2632 elif t == 'b':
2634 elif t == 'b':
2633 return "_list('%s')" % "\0".join(a.branch() for a in s)
2635 return "_list('%s')" % "\0".join(a.branch() for a in s)
2634
2636
2635 m = l // 2
2637 m = l // 2
2636 return '(%s or %s)' % (listexp(s[:m], t), listexp(s[m:], t))
2638 return '(%s or %s)' % (listexp(s[:m], t), listexp(s[m:], t))
2637
2639
2638 ret = ''
2640 ret = ''
2639 pos = 0
2641 pos = 0
2640 arg = 0
2642 arg = 0
2641 while pos < len(expr):
2643 while pos < len(expr):
2642 c = expr[pos]
2644 c = expr[pos]
2643 if c == '%':
2645 if c == '%':
2644 pos += 1
2646 pos += 1
2645 d = expr[pos]
2647 d = expr[pos]
2646 if d == '%':
2648 if d == '%':
2647 ret += d
2649 ret += d
2648 elif d in 'dsnbr':
2650 elif d in 'dsnbr':
2649 ret += argtype(d, args[arg])
2651 ret += argtype(d, args[arg])
2650 arg += 1
2652 arg += 1
2651 elif d == 'l':
2653 elif d == 'l':
2652 # a list of some type
2654 # a list of some type
2653 pos += 1
2655 pos += 1
2654 d = expr[pos]
2656 d = expr[pos]
2655 ret += listexp(list(args[arg]), d)
2657 ret += listexp(list(args[arg]), d)
2656 arg += 1
2658 arg += 1
2657 else:
2659 else:
2658 raise error.Abort('unexpected revspec format character %s' % d)
2660 raise error.Abort('unexpected revspec format character %s' % d)
2659 else:
2661 else:
2660 ret += c
2662 ret += c
2661 pos += 1
2663 pos += 1
2662
2664
2663 return ret
2665 return ret
2664
2666
2665 def prettyformat(tree):
2667 def prettyformat(tree):
2666 return parser.prettyformat(tree, ('string', 'symbol'))
2668 return parser.prettyformat(tree, ('string', 'symbol'))
2667
2669
2668 def depth(tree):
2670 def depth(tree):
2669 if isinstance(tree, tuple):
2671 if isinstance(tree, tuple):
2670 return max(map(depth, tree)) + 1
2672 return max(map(depth, tree)) + 1
2671 else:
2673 else:
2672 return 0
2674 return 0
2673
2675
2674 def funcsused(tree):
2676 def funcsused(tree):
2675 if not isinstance(tree, tuple) or tree[0] in ('string', 'symbol'):
2677 if not isinstance(tree, tuple) or tree[0] in ('string', 'symbol'):
2676 return set()
2678 return set()
2677 else:
2679 else:
2678 funcs = set()
2680 funcs = set()
2679 for s in tree[1:]:
2681 for s in tree[1:]:
2680 funcs |= funcsused(s)
2682 funcs |= funcsused(s)
2681 if tree[0] == 'func':
2683 if tree[0] == 'func':
2682 funcs.add(tree[1][1])
2684 funcs.add(tree[1][1])
2683 return funcs
2685 return funcs
2684
2686
2685 def _formatsetrepr(r):
2687 def _formatsetrepr(r):
2686 """Format an optional printable representation of a set
2688 """Format an optional printable representation of a set
2687
2689
2688 ======== =================================
2690 ======== =================================
2689 type(r) example
2691 type(r) example
2690 ======== =================================
2692 ======== =================================
2691 tuple ('<not %r>', other)
2693 tuple ('<not %r>', other)
2692 str '<branch closed>'
2694 str '<branch closed>'
2693 callable lambda: '<branch %r>' % sorted(b)
2695 callable lambda: '<branch %r>' % sorted(b)
2694 object other
2696 object other
2695 ======== =================================
2697 ======== =================================
2696 """
2698 """
2697 if r is None:
2699 if r is None:
2698 return ''
2700 return ''
2699 elif isinstance(r, tuple):
2701 elif isinstance(r, tuple):
2700 return r[0] % r[1:]
2702 return r[0] % r[1:]
2701 elif isinstance(r, str):
2703 elif isinstance(r, str):
2702 return r
2704 return r
2703 elif callable(r):
2705 elif callable(r):
2704 return r()
2706 return r()
2705 else:
2707 else:
2706 return repr(r)
2708 return repr(r)
2707
2709
2708 class abstractsmartset(object):
2710 class abstractsmartset(object):
2709
2711
2710 def __nonzero__(self):
2712 def __nonzero__(self):
2711 """True if the smartset is not empty"""
2713 """True if the smartset is not empty"""
2712 raise NotImplementedError()
2714 raise NotImplementedError()
2713
2715
2714 def __contains__(self, rev):
2716 def __contains__(self, rev):
2715 """provide fast membership testing"""
2717 """provide fast membership testing"""
2716 raise NotImplementedError()
2718 raise NotImplementedError()
2717
2719
2718 def __iter__(self):
2720 def __iter__(self):
2719 """iterate the set in the order it is supposed to be iterated"""
2721 """iterate the set in the order it is supposed to be iterated"""
2720 raise NotImplementedError()
2722 raise NotImplementedError()
2721
2723
2722 # Attributes containing a function to perform a fast iteration in a given
2724 # Attributes containing a function to perform a fast iteration in a given
2723 # direction. A smartset can have none, one, or both defined.
2725 # direction. A smartset can have none, one, or both defined.
2724 #
2726 #
2725 # Default value is None instead of a function returning None to avoid
2727 # Default value is None instead of a function returning None to avoid
2726 # initializing an iterator just for testing if a fast method exists.
2728 # initializing an iterator just for testing if a fast method exists.
2727 fastasc = None
2729 fastasc = None
2728 fastdesc = None
2730 fastdesc = None
2729
2731
2730 def isascending(self):
2732 def isascending(self):
2731 """True if the set will iterate in ascending order"""
2733 """True if the set will iterate in ascending order"""
2732 raise NotImplementedError()
2734 raise NotImplementedError()
2733
2735
2734 def isdescending(self):
2736 def isdescending(self):
2735 """True if the set will iterate in descending order"""
2737 """True if the set will iterate in descending order"""
2736 raise NotImplementedError()
2738 raise NotImplementedError()
2737
2739
2738 def istopo(self):
2740 def istopo(self):
2739 """True if the set will iterate in topographical order"""
2741 """True if the set will iterate in topographical order"""
2740 raise NotImplementedError()
2742 raise NotImplementedError()
2741
2743
2742 @util.cachefunc
2744 @util.cachefunc
2743 def min(self):
2745 def min(self):
2744 """return the minimum element in the set"""
2746 """return the minimum element in the set"""
2745 if self.fastasc is not None:
2747 if self.fastasc is not None:
2746 for r in self.fastasc():
2748 for r in self.fastasc():
2747 return r
2749 return r
2748 raise ValueError('arg is an empty sequence')
2750 raise ValueError('arg is an empty sequence')
2749 return min(self)
2751 return min(self)
2750
2752
2751 @util.cachefunc
2753 @util.cachefunc
2752 def max(self):
2754 def max(self):
2753 """return the maximum element in the set"""
2755 """return the maximum element in the set"""
2754 if self.fastdesc is not None:
2756 if self.fastdesc is not None:
2755 for r in self.fastdesc():
2757 for r in self.fastdesc():
2756 return r
2758 return r
2757 raise ValueError('arg is an empty sequence')
2759 raise ValueError('arg is an empty sequence')
2758 return max(self)
2760 return max(self)
2759
2761
2760 def first(self):
2762 def first(self):
2761 """return the first element in the set (user iteration perspective)
2763 """return the first element in the set (user iteration perspective)
2762
2764
2763 Return None if the set is empty"""
2765 Return None if the set is empty"""
2764 raise NotImplementedError()
2766 raise NotImplementedError()
2765
2767
2766 def last(self):
2768 def last(self):
2767 """return the last element in the set (user iteration perspective)
2769 """return the last element in the set (user iteration perspective)
2768
2770
2769 Return None if the set is empty"""
2771 Return None if the set is empty"""
2770 raise NotImplementedError()
2772 raise NotImplementedError()
2771
2773
2772 def __len__(self):
2774 def __len__(self):
2773 """return the length of the smartsets
2775 """return the length of the smartsets
2774
2776
2775 This can be expensive on smartset that could be lazy otherwise."""
2777 This can be expensive on smartset that could be lazy otherwise."""
2776 raise NotImplementedError()
2778 raise NotImplementedError()
2777
2779
2778 def reverse(self):
2780 def reverse(self):
2779 """reverse the expected iteration order"""
2781 """reverse the expected iteration order"""
2780 raise NotImplementedError()
2782 raise NotImplementedError()
2781
2783
2782 def sort(self, reverse=True):
2784 def sort(self, reverse=True):
2783 """get the set to iterate in an ascending or descending order"""
2785 """get the set to iterate in an ascending or descending order"""
2784 raise NotImplementedError()
2786 raise NotImplementedError()
2785
2787
2786 def __and__(self, other):
2788 def __and__(self, other):
2787 """Returns a new object with the intersection of the two collections.
2789 """Returns a new object with the intersection of the two collections.
2788
2790
2789 This is part of the mandatory API for smartset."""
2791 This is part of the mandatory API for smartset."""
2790 if isinstance(other, fullreposet):
2792 if isinstance(other, fullreposet):
2791 return self
2793 return self
2792 return self.filter(other.__contains__, condrepr=other, cache=False)
2794 return self.filter(other.__contains__, condrepr=other, cache=False)
2793
2795
2794 def __add__(self, other):
2796 def __add__(self, other):
2795 """Returns a new object with the union of the two collections.
2797 """Returns a new object with the union of the two collections.
2796
2798
2797 This is part of the mandatory API for smartset."""
2799 This is part of the mandatory API for smartset."""
2798 return addset(self, other)
2800 return addset(self, other)
2799
2801
2800 def __sub__(self, other):
2802 def __sub__(self, other):
2801 """Returns a new object with the substraction of the two collections.
2803 """Returns a new object with the substraction of the two collections.
2802
2804
2803 This is part of the mandatory API for smartset."""
2805 This is part of the mandatory API for smartset."""
2804 c = other.__contains__
2806 c = other.__contains__
2805 return self.filter(lambda r: not c(r), condrepr=('<not %r>', other),
2807 return self.filter(lambda r: not c(r), condrepr=('<not %r>', other),
2806 cache=False)
2808 cache=False)
2807
2809
2808 def filter(self, condition, condrepr=None, cache=True):
2810 def filter(self, condition, condrepr=None, cache=True):
2809 """Returns this smartset filtered by condition as a new smartset.
2811 """Returns this smartset filtered by condition as a new smartset.
2810
2812
2811 `condition` is a callable which takes a revision number and returns a
2813 `condition` is a callable which takes a revision number and returns a
2812 boolean. Optional `condrepr` provides a printable representation of
2814 boolean. Optional `condrepr` provides a printable representation of
2813 the given `condition`.
2815 the given `condition`.
2814
2816
2815 This is part of the mandatory API for smartset."""
2817 This is part of the mandatory API for smartset."""
2816 # builtin cannot be cached. but do not needs to
2818 # builtin cannot be cached. but do not needs to
2817 if cache and util.safehasattr(condition, 'func_code'):
2819 if cache and util.safehasattr(condition, 'func_code'):
2818 condition = util.cachefunc(condition)
2820 condition = util.cachefunc(condition)
2819 return filteredset(self, condition, condrepr)
2821 return filteredset(self, condition, condrepr)
2820
2822
2821 class baseset(abstractsmartset):
2823 class baseset(abstractsmartset):
2822 """Basic data structure that represents a revset and contains the basic
2824 """Basic data structure that represents a revset and contains the basic
2823 operation that it should be able to perform.
2825 operation that it should be able to perform.
2824
2826
2825 Every method in this class should be implemented by any smartset class.
2827 Every method in this class should be implemented by any smartset class.
2826 """
2828 """
2827 def __init__(self, data=(), datarepr=None, istopo=False):
2829 def __init__(self, data=(), datarepr=None, istopo=False):
2828 """
2830 """
2829 datarepr: a tuple of (format, obj, ...), a function or an object that
2831 datarepr: a tuple of (format, obj, ...), a function or an object that
2830 provides a printable representation of the given data.
2832 provides a printable representation of the given data.
2831 """
2833 """
2832 self._ascending = None
2834 self._ascending = None
2833 self._istopo = istopo
2835 self._istopo = istopo
2834 if not isinstance(data, list):
2836 if not isinstance(data, list):
2835 if isinstance(data, set):
2837 if isinstance(data, set):
2836 self._set = data
2838 self._set = data
2837 # set has no order we pick one for stability purpose
2839 # set has no order we pick one for stability purpose
2838 self._ascending = True
2840 self._ascending = True
2839 data = list(data)
2841 data = list(data)
2840 self._list = data
2842 self._list = data
2841 self._datarepr = datarepr
2843 self._datarepr = datarepr
2842
2844
2843 @util.propertycache
2845 @util.propertycache
2844 def _set(self):
2846 def _set(self):
2845 return set(self._list)
2847 return set(self._list)
2846
2848
2847 @util.propertycache
2849 @util.propertycache
2848 def _asclist(self):
2850 def _asclist(self):
2849 asclist = self._list[:]
2851 asclist = self._list[:]
2850 asclist.sort()
2852 asclist.sort()
2851 return asclist
2853 return asclist
2852
2854
2853 def __iter__(self):
2855 def __iter__(self):
2854 if self._ascending is None:
2856 if self._ascending is None:
2855 return iter(self._list)
2857 return iter(self._list)
2856 elif self._ascending:
2858 elif self._ascending:
2857 return iter(self._asclist)
2859 return iter(self._asclist)
2858 else:
2860 else:
2859 return reversed(self._asclist)
2861 return reversed(self._asclist)
2860
2862
2861 def fastasc(self):
2863 def fastasc(self):
2862 return iter(self._asclist)
2864 return iter(self._asclist)
2863
2865
2864 def fastdesc(self):
2866 def fastdesc(self):
2865 return reversed(self._asclist)
2867 return reversed(self._asclist)
2866
2868
2867 @util.propertycache
2869 @util.propertycache
2868 def __contains__(self):
2870 def __contains__(self):
2869 return self._set.__contains__
2871 return self._set.__contains__
2870
2872
2871 def __nonzero__(self):
2873 def __nonzero__(self):
2872 return bool(self._list)
2874 return bool(self._list)
2873
2875
2874 def sort(self, reverse=False):
2876 def sort(self, reverse=False):
2875 self._ascending = not bool(reverse)
2877 self._ascending = not bool(reverse)
2876 self._istopo = False
2878 self._istopo = False
2877
2879
2878 def reverse(self):
2880 def reverse(self):
2879 if self._ascending is None:
2881 if self._ascending is None:
2880 self._list.reverse()
2882 self._list.reverse()
2881 else:
2883 else:
2882 self._ascending = not self._ascending
2884 self._ascending = not self._ascending
2883 self._istopo = False
2885 self._istopo = False
2884
2886
2885 def __len__(self):
2887 def __len__(self):
2886 return len(self._list)
2888 return len(self._list)
2887
2889
2888 def isascending(self):
2890 def isascending(self):
2889 """Returns True if the collection is ascending order, False if not.
2891 """Returns True if the collection is ascending order, False if not.
2890
2892
2891 This is part of the mandatory API for smartset."""
2893 This is part of the mandatory API for smartset."""
2892 if len(self) <= 1:
2894 if len(self) <= 1:
2893 return True
2895 return True
2894 return self._ascending is not None and self._ascending
2896 return self._ascending is not None and self._ascending
2895
2897
2896 def isdescending(self):
2898 def isdescending(self):
2897 """Returns True if the collection is descending order, False if not.
2899 """Returns True if the collection is descending order, False if not.
2898
2900
2899 This is part of the mandatory API for smartset."""
2901 This is part of the mandatory API for smartset."""
2900 if len(self) <= 1:
2902 if len(self) <= 1:
2901 return True
2903 return True
2902 return self._ascending is not None and not self._ascending
2904 return self._ascending is not None and not self._ascending
2903
2905
2904 def istopo(self):
2906 def istopo(self):
2905 """Is the collection is in topographical order or not.
2907 """Is the collection is in topographical order or not.
2906
2908
2907 This is part of the mandatory API for smartset."""
2909 This is part of the mandatory API for smartset."""
2908 if len(self) <= 1:
2910 if len(self) <= 1:
2909 return True
2911 return True
2910 return self._istopo
2912 return self._istopo
2911
2913
2912 def first(self):
2914 def first(self):
2913 if self:
2915 if self:
2914 if self._ascending is None:
2916 if self._ascending is None:
2915 return self._list[0]
2917 return self._list[0]
2916 elif self._ascending:
2918 elif self._ascending:
2917 return self._asclist[0]
2919 return self._asclist[0]
2918 else:
2920 else:
2919 return self._asclist[-1]
2921 return self._asclist[-1]
2920 return None
2922 return None
2921
2923
2922 def last(self):
2924 def last(self):
2923 if self:
2925 if self:
2924 if self._ascending is None:
2926 if self._ascending is None:
2925 return self._list[-1]
2927 return self._list[-1]
2926 elif self._ascending:
2928 elif self._ascending:
2927 return self._asclist[-1]
2929 return self._asclist[-1]
2928 else:
2930 else:
2929 return self._asclist[0]
2931 return self._asclist[0]
2930 return None
2932 return None
2931
2933
2932 def __repr__(self):
2934 def __repr__(self):
2933 d = {None: '', False: '-', True: '+'}[self._ascending]
2935 d = {None: '', False: '-', True: '+'}[self._ascending]
2934 s = _formatsetrepr(self._datarepr)
2936 s = _formatsetrepr(self._datarepr)
2935 if not s:
2937 if not s:
2936 l = self._list
2938 l = self._list
2937 # if _list has been built from a set, it might have a different
2939 # if _list has been built from a set, it might have a different
2938 # order from one python implementation to another.
2940 # order from one python implementation to another.
2939 # We fallback to the sorted version for a stable output.
2941 # We fallback to the sorted version for a stable output.
2940 if self._ascending is not None:
2942 if self._ascending is not None:
2941 l = self._asclist
2943 l = self._asclist
2942 s = repr(l)
2944 s = repr(l)
2943 return '<%s%s %s>' % (type(self).__name__, d, s)
2945 return '<%s%s %s>' % (type(self).__name__, d, s)
2944
2946
2945 class filteredset(abstractsmartset):
2947 class filteredset(abstractsmartset):
2946 """Duck type for baseset class which iterates lazily over the revisions in
2948 """Duck type for baseset class which iterates lazily over the revisions in
2947 the subset and contains a function which tests for membership in the
2949 the subset and contains a function which tests for membership in the
2948 revset
2950 revset
2949 """
2951 """
2950 def __init__(self, subset, condition=lambda x: True, condrepr=None):
2952 def __init__(self, subset, condition=lambda x: True, condrepr=None):
2951 """
2953 """
2952 condition: a function that decide whether a revision in the subset
2954 condition: a function that decide whether a revision in the subset
2953 belongs to the revset or not.
2955 belongs to the revset or not.
2954 condrepr: a tuple of (format, obj, ...), a function or an object that
2956 condrepr: a tuple of (format, obj, ...), a function or an object that
2955 provides a printable representation of the given condition.
2957 provides a printable representation of the given condition.
2956 """
2958 """
2957 self._subset = subset
2959 self._subset = subset
2958 self._condition = condition
2960 self._condition = condition
2959 self._condrepr = condrepr
2961 self._condrepr = condrepr
2960
2962
2961 def __contains__(self, x):
2963 def __contains__(self, x):
2962 return x in self._subset and self._condition(x)
2964 return x in self._subset and self._condition(x)
2963
2965
2964 def __iter__(self):
2966 def __iter__(self):
2965 return self._iterfilter(self._subset)
2967 return self._iterfilter(self._subset)
2966
2968
2967 def _iterfilter(self, it):
2969 def _iterfilter(self, it):
2968 cond = self._condition
2970 cond = self._condition
2969 for x in it:
2971 for x in it:
2970 if cond(x):
2972 if cond(x):
2971 yield x
2973 yield x
2972
2974
2973 @property
2975 @property
2974 def fastasc(self):
2976 def fastasc(self):
2975 it = self._subset.fastasc
2977 it = self._subset.fastasc
2976 if it is None:
2978 if it is None:
2977 return None
2979 return None
2978 return lambda: self._iterfilter(it())
2980 return lambda: self._iterfilter(it())
2979
2981
2980 @property
2982 @property
2981 def fastdesc(self):
2983 def fastdesc(self):
2982 it = self._subset.fastdesc
2984 it = self._subset.fastdesc
2983 if it is None:
2985 if it is None:
2984 return None
2986 return None
2985 return lambda: self._iterfilter(it())
2987 return lambda: self._iterfilter(it())
2986
2988
2987 def __nonzero__(self):
2989 def __nonzero__(self):
2988 fast = None
2990 fast = None
2989 candidates = [self.fastasc if self.isascending() else None,
2991 candidates = [self.fastasc if self.isascending() else None,
2990 self.fastdesc if self.isdescending() else None,
2992 self.fastdesc if self.isdescending() else None,
2991 self.fastasc,
2993 self.fastasc,
2992 self.fastdesc]
2994 self.fastdesc]
2993 for candidate in candidates:
2995 for candidate in candidates:
2994 if candidate is not None:
2996 if candidate is not None:
2995 fast = candidate
2997 fast = candidate
2996 break
2998 break
2997
2999
2998 if fast is not None:
3000 if fast is not None:
2999 it = fast()
3001 it = fast()
3000 else:
3002 else:
3001 it = self
3003 it = self
3002
3004
3003 for r in it:
3005 for r in it:
3004 return True
3006 return True
3005 return False
3007 return False
3006
3008
3007 def __len__(self):
3009 def __len__(self):
3008 # Basic implementation to be changed in future patches.
3010 # Basic implementation to be changed in future patches.
3009 # until this gets improved, we use generator expression
3011 # until this gets improved, we use generator expression
3010 # here, since list compr is free to call __len__ again
3012 # here, since list compr is free to call __len__ again
3011 # causing infinite recursion
3013 # causing infinite recursion
3012 l = baseset(r for r in self)
3014 l = baseset(r for r in self)
3013 return len(l)
3015 return len(l)
3014
3016
3015 def sort(self, reverse=False):
3017 def sort(self, reverse=False):
3016 self._subset.sort(reverse=reverse)
3018 self._subset.sort(reverse=reverse)
3017
3019
3018 def reverse(self):
3020 def reverse(self):
3019 self._subset.reverse()
3021 self._subset.reverse()
3020
3022
3021 def isascending(self):
3023 def isascending(self):
3022 return self._subset.isascending()
3024 return self._subset.isascending()
3023
3025
3024 def isdescending(self):
3026 def isdescending(self):
3025 return self._subset.isdescending()
3027 return self._subset.isdescending()
3026
3028
3027 def istopo(self):
3029 def istopo(self):
3028 return self._subset.istopo()
3030 return self._subset.istopo()
3029
3031
3030 def first(self):
3032 def first(self):
3031 for x in self:
3033 for x in self:
3032 return x
3034 return x
3033 return None
3035 return None
3034
3036
3035 def last(self):
3037 def last(self):
3036 it = None
3038 it = None
3037 if self.isascending():
3039 if self.isascending():
3038 it = self.fastdesc
3040 it = self.fastdesc
3039 elif self.isdescending():
3041 elif self.isdescending():
3040 it = self.fastasc
3042 it = self.fastasc
3041 if it is not None:
3043 if it is not None:
3042 for x in it():
3044 for x in it():
3043 return x
3045 return x
3044 return None #empty case
3046 return None #empty case
3045 else:
3047 else:
3046 x = None
3048 x = None
3047 for x in self:
3049 for x in self:
3048 pass
3050 pass
3049 return x
3051 return x
3050
3052
3051 def __repr__(self):
3053 def __repr__(self):
3052 xs = [repr(self._subset)]
3054 xs = [repr(self._subset)]
3053 s = _formatsetrepr(self._condrepr)
3055 s = _formatsetrepr(self._condrepr)
3054 if s:
3056 if s:
3055 xs.append(s)
3057 xs.append(s)
3056 return '<%s %s>' % (type(self).__name__, ', '.join(xs))
3058 return '<%s %s>' % (type(self).__name__, ', '.join(xs))
3057
3059
3058 def _iterordered(ascending, iter1, iter2):
3060 def _iterordered(ascending, iter1, iter2):
3059 """produce an ordered iteration from two iterators with the same order
3061 """produce an ordered iteration from two iterators with the same order
3060
3062
3061 The ascending is used to indicated the iteration direction.
3063 The ascending is used to indicated the iteration direction.
3062 """
3064 """
3063 choice = max
3065 choice = max
3064 if ascending:
3066 if ascending:
3065 choice = min
3067 choice = min
3066
3068
3067 val1 = None
3069 val1 = None
3068 val2 = None
3070 val2 = None
3069 try:
3071 try:
3070 # Consume both iterators in an ordered way until one is empty
3072 # Consume both iterators in an ordered way until one is empty
3071 while True:
3073 while True:
3072 if val1 is None:
3074 if val1 is None:
3073 val1 = next(iter1)
3075 val1 = next(iter1)
3074 if val2 is None:
3076 if val2 is None:
3075 val2 = next(iter2)
3077 val2 = next(iter2)
3076 n = choice(val1, val2)
3078 n = choice(val1, val2)
3077 yield n
3079 yield n
3078 if val1 == n:
3080 if val1 == n:
3079 val1 = None
3081 val1 = None
3080 if val2 == n:
3082 if val2 == n:
3081 val2 = None
3083 val2 = None
3082 except StopIteration:
3084 except StopIteration:
3083 # Flush any remaining values and consume the other one
3085 # Flush any remaining values and consume the other one
3084 it = iter2
3086 it = iter2
3085 if val1 is not None:
3087 if val1 is not None:
3086 yield val1
3088 yield val1
3087 it = iter1
3089 it = iter1
3088 elif val2 is not None:
3090 elif val2 is not None:
3089 # might have been equality and both are empty
3091 # might have been equality and both are empty
3090 yield val2
3092 yield val2
3091 for val in it:
3093 for val in it:
3092 yield val
3094 yield val
3093
3095
3094 class addset(abstractsmartset):
3096 class addset(abstractsmartset):
3095 """Represent the addition of two sets
3097 """Represent the addition of two sets
3096
3098
3097 Wrapper structure for lazily adding two structures without losing much
3099 Wrapper structure for lazily adding two structures without losing much
3098 performance on the __contains__ method
3100 performance on the __contains__ method
3099
3101
3100 If the ascending attribute is set, that means the two structures are
3102 If the ascending attribute is set, that means the two structures are
3101 ordered in either an ascending or descending way. Therefore, we can add
3103 ordered in either an ascending or descending way. Therefore, we can add
3102 them maintaining the order by iterating over both at the same time
3104 them maintaining the order by iterating over both at the same time
3103
3105
3104 >>> xs = baseset([0, 3, 2])
3106 >>> xs = baseset([0, 3, 2])
3105 >>> ys = baseset([5, 2, 4])
3107 >>> ys = baseset([5, 2, 4])
3106
3108
3107 >>> rs = addset(xs, ys)
3109 >>> rs = addset(xs, ys)
3108 >>> bool(rs), 0 in rs, 1 in rs, 5 in rs, rs.first(), rs.last()
3110 >>> bool(rs), 0 in rs, 1 in rs, 5 in rs, rs.first(), rs.last()
3109 (True, True, False, True, 0, 4)
3111 (True, True, False, True, 0, 4)
3110 >>> rs = addset(xs, baseset([]))
3112 >>> rs = addset(xs, baseset([]))
3111 >>> bool(rs), 0 in rs, 1 in rs, rs.first(), rs.last()
3113 >>> bool(rs), 0 in rs, 1 in rs, rs.first(), rs.last()
3112 (True, True, False, 0, 2)
3114 (True, True, False, 0, 2)
3113 >>> rs = addset(baseset([]), baseset([]))
3115 >>> rs = addset(baseset([]), baseset([]))
3114 >>> bool(rs), 0 in rs, rs.first(), rs.last()
3116 >>> bool(rs), 0 in rs, rs.first(), rs.last()
3115 (False, False, None, None)
3117 (False, False, None, None)
3116
3118
3117 iterate unsorted:
3119 iterate unsorted:
3118 >>> rs = addset(xs, ys)
3120 >>> rs = addset(xs, ys)
3119 >>> # (use generator because pypy could call len())
3121 >>> # (use generator because pypy could call len())
3120 >>> list(x for x in rs) # without _genlist
3122 >>> list(x for x in rs) # without _genlist
3121 [0, 3, 2, 5, 4]
3123 [0, 3, 2, 5, 4]
3122 >>> assert not rs._genlist
3124 >>> assert not rs._genlist
3123 >>> len(rs)
3125 >>> len(rs)
3124 5
3126 5
3125 >>> [x for x in rs] # with _genlist
3127 >>> [x for x in rs] # with _genlist
3126 [0, 3, 2, 5, 4]
3128 [0, 3, 2, 5, 4]
3127 >>> assert rs._genlist
3129 >>> assert rs._genlist
3128
3130
3129 iterate ascending:
3131 iterate ascending:
3130 >>> rs = addset(xs, ys, ascending=True)
3132 >>> rs = addset(xs, ys, ascending=True)
3131 >>> # (use generator because pypy could call len())
3133 >>> # (use generator because pypy could call len())
3132 >>> list(x for x in rs), list(x for x in rs.fastasc()) # without _asclist
3134 >>> list(x for x in rs), list(x for x in rs.fastasc()) # without _asclist
3133 ([0, 2, 3, 4, 5], [0, 2, 3, 4, 5])
3135 ([0, 2, 3, 4, 5], [0, 2, 3, 4, 5])
3134 >>> assert not rs._asclist
3136 >>> assert not rs._asclist
3135 >>> len(rs)
3137 >>> len(rs)
3136 5
3138 5
3137 >>> [x for x in rs], [x for x in rs.fastasc()]
3139 >>> [x for x in rs], [x for x in rs.fastasc()]
3138 ([0, 2, 3, 4, 5], [0, 2, 3, 4, 5])
3140 ([0, 2, 3, 4, 5], [0, 2, 3, 4, 5])
3139 >>> assert rs._asclist
3141 >>> assert rs._asclist
3140
3142
3141 iterate descending:
3143 iterate descending:
3142 >>> rs = addset(xs, ys, ascending=False)
3144 >>> rs = addset(xs, ys, ascending=False)
3143 >>> # (use generator because pypy could call len())
3145 >>> # (use generator because pypy could call len())
3144 >>> list(x for x in rs), list(x for x in rs.fastdesc()) # without _asclist
3146 >>> list(x for x in rs), list(x for x in rs.fastdesc()) # without _asclist
3145 ([5, 4, 3, 2, 0], [5, 4, 3, 2, 0])
3147 ([5, 4, 3, 2, 0], [5, 4, 3, 2, 0])
3146 >>> assert not rs._asclist
3148 >>> assert not rs._asclist
3147 >>> len(rs)
3149 >>> len(rs)
3148 5
3150 5
3149 >>> [x for x in rs], [x for x in rs.fastdesc()]
3151 >>> [x for x in rs], [x for x in rs.fastdesc()]
3150 ([5, 4, 3, 2, 0], [5, 4, 3, 2, 0])
3152 ([5, 4, 3, 2, 0], [5, 4, 3, 2, 0])
3151 >>> assert rs._asclist
3153 >>> assert rs._asclist
3152
3154
3153 iterate ascending without fastasc:
3155 iterate ascending without fastasc:
3154 >>> rs = addset(xs, generatorset(ys), ascending=True)
3156 >>> rs = addset(xs, generatorset(ys), ascending=True)
3155 >>> assert rs.fastasc is None
3157 >>> assert rs.fastasc is None
3156 >>> [x for x in rs]
3158 >>> [x for x in rs]
3157 [0, 2, 3, 4, 5]
3159 [0, 2, 3, 4, 5]
3158
3160
3159 iterate descending without fastdesc:
3161 iterate descending without fastdesc:
3160 >>> rs = addset(generatorset(xs), ys, ascending=False)
3162 >>> rs = addset(generatorset(xs), ys, ascending=False)
3161 >>> assert rs.fastdesc is None
3163 >>> assert rs.fastdesc is None
3162 >>> [x for x in rs]
3164 >>> [x for x in rs]
3163 [5, 4, 3, 2, 0]
3165 [5, 4, 3, 2, 0]
3164 """
3166 """
3165 def __init__(self, revs1, revs2, ascending=None):
3167 def __init__(self, revs1, revs2, ascending=None):
3166 self._r1 = revs1
3168 self._r1 = revs1
3167 self._r2 = revs2
3169 self._r2 = revs2
3168 self._iter = None
3170 self._iter = None
3169 self._ascending = ascending
3171 self._ascending = ascending
3170 self._genlist = None
3172 self._genlist = None
3171 self._asclist = None
3173 self._asclist = None
3172
3174
3173 def __len__(self):
3175 def __len__(self):
3174 return len(self._list)
3176 return len(self._list)
3175
3177
3176 def __nonzero__(self):
3178 def __nonzero__(self):
3177 return bool(self._r1) or bool(self._r2)
3179 return bool(self._r1) or bool(self._r2)
3178
3180
3179 @util.propertycache
3181 @util.propertycache
3180 def _list(self):
3182 def _list(self):
3181 if not self._genlist:
3183 if not self._genlist:
3182 self._genlist = baseset(iter(self))
3184 self._genlist = baseset(iter(self))
3183 return self._genlist
3185 return self._genlist
3184
3186
3185 def __iter__(self):
3187 def __iter__(self):
3186 """Iterate over both collections without repeating elements
3188 """Iterate over both collections without repeating elements
3187
3189
3188 If the ascending attribute is not set, iterate over the first one and
3190 If the ascending attribute is not set, iterate over the first one and
3189 then over the second one checking for membership on the first one so we
3191 then over the second one checking for membership on the first one so we
3190 dont yield any duplicates.
3192 dont yield any duplicates.
3191
3193
3192 If the ascending attribute is set, iterate over both collections at the
3194 If the ascending attribute is set, iterate over both collections at the
3193 same time, yielding only one value at a time in the given order.
3195 same time, yielding only one value at a time in the given order.
3194 """
3196 """
3195 if self._ascending is None:
3197 if self._ascending is None:
3196 if self._genlist:
3198 if self._genlist:
3197 return iter(self._genlist)
3199 return iter(self._genlist)
3198 def arbitraryordergen():
3200 def arbitraryordergen():
3199 for r in self._r1:
3201 for r in self._r1:
3200 yield r
3202 yield r
3201 inr1 = self._r1.__contains__
3203 inr1 = self._r1.__contains__
3202 for r in self._r2:
3204 for r in self._r2:
3203 if not inr1(r):
3205 if not inr1(r):
3204 yield r
3206 yield r
3205 return arbitraryordergen()
3207 return arbitraryordergen()
3206 # try to use our own fast iterator if it exists
3208 # try to use our own fast iterator if it exists
3207 self._trysetasclist()
3209 self._trysetasclist()
3208 if self._ascending:
3210 if self._ascending:
3209 attr = 'fastasc'
3211 attr = 'fastasc'
3210 else:
3212 else:
3211 attr = 'fastdesc'
3213 attr = 'fastdesc'
3212 it = getattr(self, attr)
3214 it = getattr(self, attr)
3213 if it is not None:
3215 if it is not None:
3214 return it()
3216 return it()
3215 # maybe half of the component supports fast
3217 # maybe half of the component supports fast
3216 # get iterator for _r1
3218 # get iterator for _r1
3217 iter1 = getattr(self._r1, attr)
3219 iter1 = getattr(self._r1, attr)
3218 if iter1 is None:
3220 if iter1 is None:
3219 # let's avoid side effect (not sure it matters)
3221 # let's avoid side effect (not sure it matters)
3220 iter1 = iter(sorted(self._r1, reverse=not self._ascending))
3222 iter1 = iter(sorted(self._r1, reverse=not self._ascending))
3221 else:
3223 else:
3222 iter1 = iter1()
3224 iter1 = iter1()
3223 # get iterator for _r2
3225 # get iterator for _r2
3224 iter2 = getattr(self._r2, attr)
3226 iter2 = getattr(self._r2, attr)
3225 if iter2 is None:
3227 if iter2 is None:
3226 # let's avoid side effect (not sure it matters)
3228 # let's avoid side effect (not sure it matters)
3227 iter2 = iter(sorted(self._r2, reverse=not self._ascending))
3229 iter2 = iter(sorted(self._r2, reverse=not self._ascending))
3228 else:
3230 else:
3229 iter2 = iter2()
3231 iter2 = iter2()
3230 return _iterordered(self._ascending, iter1, iter2)
3232 return _iterordered(self._ascending, iter1, iter2)
3231
3233
3232 def _trysetasclist(self):
3234 def _trysetasclist(self):
3233 """populate the _asclist attribute if possible and necessary"""
3235 """populate the _asclist attribute if possible and necessary"""
3234 if self._genlist is not None and self._asclist is None:
3236 if self._genlist is not None and self._asclist is None:
3235 self._asclist = sorted(self._genlist)
3237 self._asclist = sorted(self._genlist)
3236
3238
3237 @property
3239 @property
3238 def fastasc(self):
3240 def fastasc(self):
3239 self._trysetasclist()
3241 self._trysetasclist()
3240 if self._asclist is not None:
3242 if self._asclist is not None:
3241 return self._asclist.__iter__
3243 return self._asclist.__iter__
3242 iter1 = self._r1.fastasc
3244 iter1 = self._r1.fastasc
3243 iter2 = self._r2.fastasc
3245 iter2 = self._r2.fastasc
3244 if None in (iter1, iter2):
3246 if None in (iter1, iter2):
3245 return None
3247 return None
3246 return lambda: _iterordered(True, iter1(), iter2())
3248 return lambda: _iterordered(True, iter1(), iter2())
3247
3249
3248 @property
3250 @property
3249 def fastdesc(self):
3251 def fastdesc(self):
3250 self._trysetasclist()
3252 self._trysetasclist()
3251 if self._asclist is not None:
3253 if self._asclist is not None:
3252 return self._asclist.__reversed__
3254 return self._asclist.__reversed__
3253 iter1 = self._r1.fastdesc
3255 iter1 = self._r1.fastdesc
3254 iter2 = self._r2.fastdesc
3256 iter2 = self._r2.fastdesc
3255 if None in (iter1, iter2):
3257 if None in (iter1, iter2):
3256 return None
3258 return None
3257 return lambda: _iterordered(False, iter1(), iter2())
3259 return lambda: _iterordered(False, iter1(), iter2())
3258
3260
3259 def __contains__(self, x):
3261 def __contains__(self, x):
3260 return x in self._r1 or x in self._r2
3262 return x in self._r1 or x in self._r2
3261
3263
3262 def sort(self, reverse=False):
3264 def sort(self, reverse=False):
3263 """Sort the added set
3265 """Sort the added set
3264
3266
3265 For this we use the cached list with all the generated values and if we
3267 For this we use the cached list with all the generated values and if we
3266 know they are ascending or descending we can sort them in a smart way.
3268 know they are ascending or descending we can sort them in a smart way.
3267 """
3269 """
3268 self._ascending = not reverse
3270 self._ascending = not reverse
3269
3271
3270 def isascending(self):
3272 def isascending(self):
3271 return self._ascending is not None and self._ascending
3273 return self._ascending is not None and self._ascending
3272
3274
3273 def isdescending(self):
3275 def isdescending(self):
3274 return self._ascending is not None and not self._ascending
3276 return self._ascending is not None and not self._ascending
3275
3277
3276 def istopo(self):
3278 def istopo(self):
3277 # not worth the trouble asserting if the two sets combined are still
3279 # not worth the trouble asserting if the two sets combined are still
3278 # in topographical order. Use the sort() predicate to explicitly sort
3280 # in topographical order. Use the sort() predicate to explicitly sort
3279 # again instead.
3281 # again instead.
3280 return False
3282 return False
3281
3283
3282 def reverse(self):
3284 def reverse(self):
3283 if self._ascending is None:
3285 if self._ascending is None:
3284 self._list.reverse()
3286 self._list.reverse()
3285 else:
3287 else:
3286 self._ascending = not self._ascending
3288 self._ascending = not self._ascending
3287
3289
3288 def first(self):
3290 def first(self):
3289 for x in self:
3291 for x in self:
3290 return x
3292 return x
3291 return None
3293 return None
3292
3294
3293 def last(self):
3295 def last(self):
3294 self.reverse()
3296 self.reverse()
3295 val = self.first()
3297 val = self.first()
3296 self.reverse()
3298 self.reverse()
3297 return val
3299 return val
3298
3300
3299 def __repr__(self):
3301 def __repr__(self):
3300 d = {None: '', False: '-', True: '+'}[self._ascending]
3302 d = {None: '', False: '-', True: '+'}[self._ascending]
3301 return '<%s%s %r, %r>' % (type(self).__name__, d, self._r1, self._r2)
3303 return '<%s%s %r, %r>' % (type(self).__name__, d, self._r1, self._r2)
3302
3304
3303 class generatorset(abstractsmartset):
3305 class generatorset(abstractsmartset):
3304 """Wrap a generator for lazy iteration
3306 """Wrap a generator for lazy iteration
3305
3307
3306 Wrapper structure for generators that provides lazy membership and can
3308 Wrapper structure for generators that provides lazy membership and can
3307 be iterated more than once.
3309 be iterated more than once.
3308 When asked for membership it generates values until either it finds the
3310 When asked for membership it generates values until either it finds the
3309 requested one or has gone through all the elements in the generator
3311 requested one or has gone through all the elements in the generator
3310 """
3312 """
3311 def __init__(self, gen, iterasc=None):
3313 def __init__(self, gen, iterasc=None):
3312 """
3314 """
3313 gen: a generator producing the values for the generatorset.
3315 gen: a generator producing the values for the generatorset.
3314 """
3316 """
3315 self._gen = gen
3317 self._gen = gen
3316 self._asclist = None
3318 self._asclist = None
3317 self._cache = {}
3319 self._cache = {}
3318 self._genlist = []
3320 self._genlist = []
3319 self._finished = False
3321 self._finished = False
3320 self._ascending = True
3322 self._ascending = True
3321 if iterasc is not None:
3323 if iterasc is not None:
3322 if iterasc:
3324 if iterasc:
3323 self.fastasc = self._iterator
3325 self.fastasc = self._iterator
3324 self.__contains__ = self._asccontains
3326 self.__contains__ = self._asccontains
3325 else:
3327 else:
3326 self.fastdesc = self._iterator
3328 self.fastdesc = self._iterator
3327 self.__contains__ = self._desccontains
3329 self.__contains__ = self._desccontains
3328
3330
3329 def __nonzero__(self):
3331 def __nonzero__(self):
3330 # Do not use 'for r in self' because it will enforce the iteration
3332 # Do not use 'for r in self' because it will enforce the iteration
3331 # order (default ascending), possibly unrolling a whole descending
3333 # order (default ascending), possibly unrolling a whole descending
3332 # iterator.
3334 # iterator.
3333 if self._genlist:
3335 if self._genlist:
3334 return True
3336 return True
3335 for r in self._consumegen():
3337 for r in self._consumegen():
3336 return True
3338 return True
3337 return False
3339 return False
3338
3340
3339 def __contains__(self, x):
3341 def __contains__(self, x):
3340 if x in self._cache:
3342 if x in self._cache:
3341 return self._cache[x]
3343 return self._cache[x]
3342
3344
3343 # Use new values only, as existing values would be cached.
3345 # Use new values only, as existing values would be cached.
3344 for l in self._consumegen():
3346 for l in self._consumegen():
3345 if l == x:
3347 if l == x:
3346 return True
3348 return True
3347
3349
3348 self._cache[x] = False
3350 self._cache[x] = False
3349 return False
3351 return False
3350
3352
3351 def _asccontains(self, x):
3353 def _asccontains(self, x):
3352 """version of contains optimised for ascending generator"""
3354 """version of contains optimised for ascending generator"""
3353 if x in self._cache:
3355 if x in self._cache:
3354 return self._cache[x]
3356 return self._cache[x]
3355
3357
3356 # Use new values only, as existing values would be cached.
3358 # Use new values only, as existing values would be cached.
3357 for l in self._consumegen():
3359 for l in self._consumegen():
3358 if l == x:
3360 if l == x:
3359 return True
3361 return True
3360 if l > x:
3362 if l > x:
3361 break
3363 break
3362
3364
3363 self._cache[x] = False
3365 self._cache[x] = False
3364 return False
3366 return False
3365
3367
3366 def _desccontains(self, x):
3368 def _desccontains(self, x):
3367 """version of contains optimised for descending generator"""
3369 """version of contains optimised for descending generator"""
3368 if x in self._cache:
3370 if x in self._cache:
3369 return self._cache[x]
3371 return self._cache[x]
3370
3372
3371 # Use new values only, as existing values would be cached.
3373 # Use new values only, as existing values would be cached.
3372 for l in self._consumegen():
3374 for l in self._consumegen():
3373 if l == x:
3375 if l == x:
3374 return True
3376 return True
3375 if l < x:
3377 if l < x:
3376 break
3378 break
3377
3379
3378 self._cache[x] = False
3380 self._cache[x] = False
3379 return False
3381 return False
3380
3382
3381 def __iter__(self):
3383 def __iter__(self):
3382 if self._ascending:
3384 if self._ascending:
3383 it = self.fastasc
3385 it = self.fastasc
3384 else:
3386 else:
3385 it = self.fastdesc
3387 it = self.fastdesc
3386 if it is not None:
3388 if it is not None:
3387 return it()
3389 return it()
3388 # we need to consume the iterator
3390 # we need to consume the iterator
3389 for x in self._consumegen():
3391 for x in self._consumegen():
3390 pass
3392 pass
3391 # recall the same code
3393 # recall the same code
3392 return iter(self)
3394 return iter(self)
3393
3395
3394 def _iterator(self):
3396 def _iterator(self):
3395 if self._finished:
3397 if self._finished:
3396 return iter(self._genlist)
3398 return iter(self._genlist)
3397
3399
3398 # We have to use this complex iteration strategy to allow multiple
3400 # We have to use this complex iteration strategy to allow multiple
3399 # iterations at the same time. We need to be able to catch revision
3401 # iterations at the same time. We need to be able to catch revision
3400 # removed from _consumegen and added to genlist in another instance.
3402 # removed from _consumegen and added to genlist in another instance.
3401 #
3403 #
3402 # Getting rid of it would provide an about 15% speed up on this
3404 # Getting rid of it would provide an about 15% speed up on this
3403 # iteration.
3405 # iteration.
3404 genlist = self._genlist
3406 genlist = self._genlist
3405 nextrev = self._consumegen().next
3407 nextrev = self._consumegen().next
3406 _len = len # cache global lookup
3408 _len = len # cache global lookup
3407 def gen():
3409 def gen():
3408 i = 0
3410 i = 0
3409 while True:
3411 while True:
3410 if i < _len(genlist):
3412 if i < _len(genlist):
3411 yield genlist[i]
3413 yield genlist[i]
3412 else:
3414 else:
3413 yield nextrev()
3415 yield nextrev()
3414 i += 1
3416 i += 1
3415 return gen()
3417 return gen()
3416
3418
3417 def _consumegen(self):
3419 def _consumegen(self):
3418 cache = self._cache
3420 cache = self._cache
3419 genlist = self._genlist.append
3421 genlist = self._genlist.append
3420 for item in self._gen:
3422 for item in self._gen:
3421 cache[item] = True
3423 cache[item] = True
3422 genlist(item)
3424 genlist(item)
3423 yield item
3425 yield item
3424 if not self._finished:
3426 if not self._finished:
3425 self._finished = True
3427 self._finished = True
3426 asc = self._genlist[:]
3428 asc = self._genlist[:]
3427 asc.sort()
3429 asc.sort()
3428 self._asclist = asc
3430 self._asclist = asc
3429 self.fastasc = asc.__iter__
3431 self.fastasc = asc.__iter__
3430 self.fastdesc = asc.__reversed__
3432 self.fastdesc = asc.__reversed__
3431
3433
3432 def __len__(self):
3434 def __len__(self):
3433 for x in self._consumegen():
3435 for x in self._consumegen():
3434 pass
3436 pass
3435 return len(self._genlist)
3437 return len(self._genlist)
3436
3438
3437 def sort(self, reverse=False):
3439 def sort(self, reverse=False):
3438 self._ascending = not reverse
3440 self._ascending = not reverse
3439
3441
3440 def reverse(self):
3442 def reverse(self):
3441 self._ascending = not self._ascending
3443 self._ascending = not self._ascending
3442
3444
3443 def isascending(self):
3445 def isascending(self):
3444 return self._ascending
3446 return self._ascending
3445
3447
3446 def isdescending(self):
3448 def isdescending(self):
3447 return not self._ascending
3449 return not self._ascending
3448
3450
3449 def istopo(self):
3451 def istopo(self):
3450 # not worth the trouble asserting if the two sets combined are still
3452 # not worth the trouble asserting if the two sets combined are still
3451 # in topographical order. Use the sort() predicate to explicitly sort
3453 # in topographical order. Use the sort() predicate to explicitly sort
3452 # again instead.
3454 # again instead.
3453 return False
3455 return False
3454
3456
3455 def first(self):
3457 def first(self):
3456 if self._ascending:
3458 if self._ascending:
3457 it = self.fastasc
3459 it = self.fastasc
3458 else:
3460 else:
3459 it = self.fastdesc
3461 it = self.fastdesc
3460 if it is None:
3462 if it is None:
3461 # we need to consume all and try again
3463 # we need to consume all and try again
3462 for x in self._consumegen():
3464 for x in self._consumegen():
3463 pass
3465 pass
3464 return self.first()
3466 return self.first()
3465 return next(it(), None)
3467 return next(it(), None)
3466
3468
3467 def last(self):
3469 def last(self):
3468 if self._ascending:
3470 if self._ascending:
3469 it = self.fastdesc
3471 it = self.fastdesc
3470 else:
3472 else:
3471 it = self.fastasc
3473 it = self.fastasc
3472 if it is None:
3474 if it is None:
3473 # we need to consume all and try again
3475 # we need to consume all and try again
3474 for x in self._consumegen():
3476 for x in self._consumegen():
3475 pass
3477 pass
3476 return self.first()
3478 return self.first()
3477 return next(it(), None)
3479 return next(it(), None)
3478
3480
3479 def __repr__(self):
3481 def __repr__(self):
3480 d = {False: '-', True: '+'}[self._ascending]
3482 d = {False: '-', True: '+'}[self._ascending]
3481 return '<%s%s>' % (type(self).__name__, d)
3483 return '<%s%s>' % (type(self).__name__, d)
3482
3484
3483 class spanset(abstractsmartset):
3485 class spanset(abstractsmartset):
3484 """Duck type for baseset class which represents a range of revisions and
3486 """Duck type for baseset class which represents a range of revisions and
3485 can work lazily and without having all the range in memory
3487 can work lazily and without having all the range in memory
3486
3488
3487 Note that spanset(x, y) behave almost like xrange(x, y) except for two
3489 Note that spanset(x, y) behave almost like xrange(x, y) except for two
3488 notable points:
3490 notable points:
3489 - when x < y it will be automatically descending,
3491 - when x < y it will be automatically descending,
3490 - revision filtered with this repoview will be skipped.
3492 - revision filtered with this repoview will be skipped.
3491
3493
3492 """
3494 """
3493 def __init__(self, repo, start=0, end=None):
3495 def __init__(self, repo, start=0, end=None):
3494 """
3496 """
3495 start: first revision included the set
3497 start: first revision included the set
3496 (default to 0)
3498 (default to 0)
3497 end: first revision excluded (last+1)
3499 end: first revision excluded (last+1)
3498 (default to len(repo)
3500 (default to len(repo)
3499
3501
3500 Spanset will be descending if `end` < `start`.
3502 Spanset will be descending if `end` < `start`.
3501 """
3503 """
3502 if end is None:
3504 if end is None:
3503 end = len(repo)
3505 end = len(repo)
3504 self._ascending = start <= end
3506 self._ascending = start <= end
3505 if not self._ascending:
3507 if not self._ascending:
3506 start, end = end + 1, start +1
3508 start, end = end + 1, start +1
3507 self._start = start
3509 self._start = start
3508 self._end = end
3510 self._end = end
3509 self._hiddenrevs = repo.changelog.filteredrevs
3511 self._hiddenrevs = repo.changelog.filteredrevs
3510
3512
3511 def sort(self, reverse=False):
3513 def sort(self, reverse=False):
3512 self._ascending = not reverse
3514 self._ascending = not reverse
3513
3515
3514 def reverse(self):
3516 def reverse(self):
3515 self._ascending = not self._ascending
3517 self._ascending = not self._ascending
3516
3518
3517 def istopo(self):
3519 def istopo(self):
3518 # not worth the trouble asserting if the two sets combined are still
3520 # not worth the trouble asserting if the two sets combined are still
3519 # in topographical order. Use the sort() predicate to explicitly sort
3521 # in topographical order. Use the sort() predicate to explicitly sort
3520 # again instead.
3522 # again instead.
3521 return False
3523 return False
3522
3524
3523 def _iterfilter(self, iterrange):
3525 def _iterfilter(self, iterrange):
3524 s = self._hiddenrevs
3526 s = self._hiddenrevs
3525 for r in iterrange:
3527 for r in iterrange:
3526 if r not in s:
3528 if r not in s:
3527 yield r
3529 yield r
3528
3530
3529 def __iter__(self):
3531 def __iter__(self):
3530 if self._ascending:
3532 if self._ascending:
3531 return self.fastasc()
3533 return self.fastasc()
3532 else:
3534 else:
3533 return self.fastdesc()
3535 return self.fastdesc()
3534
3536
3535 def fastasc(self):
3537 def fastasc(self):
3536 iterrange = xrange(self._start, self._end)
3538 iterrange = xrange(self._start, self._end)
3537 if self._hiddenrevs:
3539 if self._hiddenrevs:
3538 return self._iterfilter(iterrange)
3540 return self._iterfilter(iterrange)
3539 return iter(iterrange)
3541 return iter(iterrange)
3540
3542
3541 def fastdesc(self):
3543 def fastdesc(self):
3542 iterrange = xrange(self._end - 1, self._start - 1, -1)
3544 iterrange = xrange(self._end - 1, self._start - 1, -1)
3543 if self._hiddenrevs:
3545 if self._hiddenrevs:
3544 return self._iterfilter(iterrange)
3546 return self._iterfilter(iterrange)
3545 return iter(iterrange)
3547 return iter(iterrange)
3546
3548
3547 def __contains__(self, rev):
3549 def __contains__(self, rev):
3548 hidden = self._hiddenrevs
3550 hidden = self._hiddenrevs
3549 return ((self._start <= rev < self._end)
3551 return ((self._start <= rev < self._end)
3550 and not (hidden and rev in hidden))
3552 and not (hidden and rev in hidden))
3551
3553
3552 def __nonzero__(self):
3554 def __nonzero__(self):
3553 for r in self:
3555 for r in self:
3554 return True
3556 return True
3555 return False
3557 return False
3556
3558
3557 def __len__(self):
3559 def __len__(self):
3558 if not self._hiddenrevs:
3560 if not self._hiddenrevs:
3559 return abs(self._end - self._start)
3561 return abs(self._end - self._start)
3560 else:
3562 else:
3561 count = 0
3563 count = 0
3562 start = self._start
3564 start = self._start
3563 end = self._end
3565 end = self._end
3564 for rev in self._hiddenrevs:
3566 for rev in self._hiddenrevs:
3565 if (end < rev <= start) or (start <= rev < end):
3567 if (end < rev <= start) or (start <= rev < end):
3566 count += 1
3568 count += 1
3567 return abs(self._end - self._start) - count
3569 return abs(self._end - self._start) - count
3568
3570
3569 def isascending(self):
3571 def isascending(self):
3570 return self._ascending
3572 return self._ascending
3571
3573
3572 def isdescending(self):
3574 def isdescending(self):
3573 return not self._ascending
3575 return not self._ascending
3574
3576
3575 def first(self):
3577 def first(self):
3576 if self._ascending:
3578 if self._ascending:
3577 it = self.fastasc
3579 it = self.fastasc
3578 else:
3580 else:
3579 it = self.fastdesc
3581 it = self.fastdesc
3580 for x in it():
3582 for x in it():
3581 return x
3583 return x
3582 return None
3584 return None
3583
3585
3584 def last(self):
3586 def last(self):
3585 if self._ascending:
3587 if self._ascending:
3586 it = self.fastdesc
3588 it = self.fastdesc
3587 else:
3589 else:
3588 it = self.fastasc
3590 it = self.fastasc
3589 for x in it():
3591 for x in it():
3590 return x
3592 return x
3591 return None
3593 return None
3592
3594
3593 def __repr__(self):
3595 def __repr__(self):
3594 d = {False: '-', True: '+'}[self._ascending]
3596 d = {False: '-', True: '+'}[self._ascending]
3595 return '<%s%s %d:%d>' % (type(self).__name__, d,
3597 return '<%s%s %d:%d>' % (type(self).__name__, d,
3596 self._start, self._end - 1)
3598 self._start, self._end - 1)
3597
3599
3598 class fullreposet(spanset):
3600 class fullreposet(spanset):
3599 """a set containing all revisions in the repo
3601 """a set containing all revisions in the repo
3600
3602
3601 This class exists to host special optimization and magic to handle virtual
3603 This class exists to host special optimization and magic to handle virtual
3602 revisions such as "null".
3604 revisions such as "null".
3603 """
3605 """
3604
3606
3605 def __init__(self, repo):
3607 def __init__(self, repo):
3606 super(fullreposet, self).__init__(repo)
3608 super(fullreposet, self).__init__(repo)
3607
3609
3608 def __and__(self, other):
3610 def __and__(self, other):
3609 """As self contains the whole repo, all of the other set should also be
3611 """As self contains the whole repo, all of the other set should also be
3610 in self. Therefore `self & other = other`.
3612 in self. Therefore `self & other = other`.
3611
3613
3612 This boldly assumes the other contains valid revs only.
3614 This boldly assumes the other contains valid revs only.
3613 """
3615 """
3614 # other not a smartset, make is so
3616 # other not a smartset, make is so
3615 if not util.safehasattr(other, 'isascending'):
3617 if not util.safehasattr(other, 'isascending'):
3616 # filter out hidden revision
3618 # filter out hidden revision
3617 # (this boldly assumes all smartset are pure)
3619 # (this boldly assumes all smartset are pure)
3618 #
3620 #
3619 # `other` was used with "&", let's assume this is a set like
3621 # `other` was used with "&", let's assume this is a set like
3620 # object.
3622 # object.
3621 other = baseset(other - self._hiddenrevs)
3623 other = baseset(other - self._hiddenrevs)
3622
3624
3623 # XXX As fullreposet is also used as bootstrap, this is wrong.
3625 # XXX As fullreposet is also used as bootstrap, this is wrong.
3624 #
3626 #
3625 # With a giveme312() revset returning [3,1,2], this makes
3627 # With a giveme312() revset returning [3,1,2], this makes
3626 # 'hg log -r "giveme312()"' -> 1, 2, 3 (wrong)
3628 # 'hg log -r "giveme312()"' -> 1, 2, 3 (wrong)
3627 # We cannot just drop it because other usage still need to sort it:
3629 # We cannot just drop it because other usage still need to sort it:
3628 # 'hg log -r "all() and giveme312()"' -> 1, 2, 3 (right)
3630 # 'hg log -r "all() and giveme312()"' -> 1, 2, 3 (right)
3629 #
3631 #
3630 # There is also some faulty revset implementations that rely on it
3632 # There is also some faulty revset implementations that rely on it
3631 # (eg: children as of its state in e8075329c5fb)
3633 # (eg: children as of its state in e8075329c5fb)
3632 #
3634 #
3633 # When we fix the two points above we can move this into the if clause
3635 # When we fix the two points above we can move this into the if clause
3634 other.sort(reverse=self.isdescending())
3636 other.sort(reverse=self.isdescending())
3635 return other
3637 return other
3636
3638
3637 def prettyformatset(revs):
3639 def prettyformatset(revs):
3638 lines = []
3640 lines = []
3639 rs = repr(revs)
3641 rs = repr(revs)
3640 p = 0
3642 p = 0
3641 while p < len(rs):
3643 while p < len(rs):
3642 q = rs.find('<', p + 1)
3644 q = rs.find('<', p + 1)
3643 if q < 0:
3645 if q < 0:
3644 q = len(rs)
3646 q = len(rs)
3645 l = rs.count('<', 0, p) - rs.count('>', 0, p)
3647 l = rs.count('<', 0, p) - rs.count('>', 0, p)
3646 assert l >= 0
3648 assert l >= 0
3647 lines.append((l, rs[p:q].rstrip()))
3649 lines.append((l, rs[p:q].rstrip()))
3648 p = q
3650 p = q
3649 return '\n'.join(' ' * l + s for l, s in lines)
3651 return '\n'.join(' ' * l + s for l, s in lines)
3650
3652
3651 def loadpredicate(ui, extname, registrarobj):
3653 def loadpredicate(ui, extname, registrarobj):
3652 """Load revset predicates from specified registrarobj
3654 """Load revset predicates from specified registrarobj
3653 """
3655 """
3654 for name, func in registrarobj._table.iteritems():
3656 for name, func in registrarobj._table.iteritems():
3655 symbols[name] = func
3657 symbols[name] = func
3656 if func._safe:
3658 if func._safe:
3657 safesymbols.add(name)
3659 safesymbols.add(name)
3658
3660
3659 # load built-in predicates explicitly to setup safesymbols
3661 # load built-in predicates explicitly to setup safesymbols
3660 loadpredicate(None, None, predicate)
3662 loadpredicate(None, None, predicate)
3661
3663
3662 # tell hggettext to extract docstrings from these functions:
3664 # tell hggettext to extract docstrings from these functions:
3663 i18nfunctions = symbols.values()
3665 i18nfunctions = symbols.values()
@@ -1,2560 +1,2567
1 $ HGENCODING=utf-8
1 $ HGENCODING=utf-8
2 $ export HGENCODING
2 $ export HGENCODING
3 $ cat > testrevset.py << EOF
3 $ cat > testrevset.py << EOF
4 > import mercurial.revset
4 > import mercurial.revset
5 >
5 >
6 > baseset = mercurial.revset.baseset
6 > baseset = mercurial.revset.baseset
7 >
7 >
8 > def r3232(repo, subset, x):
8 > def r3232(repo, subset, x):
9 > """"simple revset that return [3,2,3,2]
9 > """"simple revset that return [3,2,3,2]
10 >
10 >
11 > revisions duplicated on purpose.
11 > revisions duplicated on purpose.
12 > """
12 > """
13 > if 3 not in subset:
13 > if 3 not in subset:
14 > if 2 in subset:
14 > if 2 in subset:
15 > return baseset([2,2])
15 > return baseset([2,2])
16 > return baseset()
16 > return baseset()
17 > return baseset([3,3,2,2])
17 > return baseset([3,3,2,2])
18 >
18 >
19 > mercurial.revset.symbols['r3232'] = r3232
19 > mercurial.revset.symbols['r3232'] = r3232
20 > EOF
20 > EOF
21 $ cat >> $HGRCPATH << EOF
21 $ cat >> $HGRCPATH << EOF
22 > [extensions]
22 > [extensions]
23 > testrevset=$TESTTMP/testrevset.py
23 > testrevset=$TESTTMP/testrevset.py
24 > EOF
24 > EOF
25
25
26 $ try() {
26 $ try() {
27 > hg debugrevspec --debug "$@"
27 > hg debugrevspec --debug "$@"
28 > }
28 > }
29
29
30 $ log() {
30 $ log() {
31 > hg log --template '{rev}\n' -r "$1"
31 > hg log --template '{rev}\n' -r "$1"
32 > }
32 > }
33
33
34 $ hg init repo
34 $ hg init repo
35 $ cd repo
35 $ cd repo
36
36
37 $ echo a > a
37 $ echo a > a
38 $ hg branch a
38 $ hg branch a
39 marked working directory as branch a
39 marked working directory as branch a
40 (branches are permanent and global, did you want a bookmark?)
40 (branches are permanent and global, did you want a bookmark?)
41 $ hg ci -Aqm0
41 $ hg ci -Aqm0
42
42
43 $ echo b > b
43 $ echo b > b
44 $ hg branch b
44 $ hg branch b
45 marked working directory as branch b
45 marked working directory as branch b
46 $ hg ci -Aqm1
46 $ hg ci -Aqm1
47
47
48 $ rm a
48 $ rm a
49 $ hg branch a-b-c-
49 $ hg branch a-b-c-
50 marked working directory as branch a-b-c-
50 marked working directory as branch a-b-c-
51 $ hg ci -Aqm2 -u Bob
51 $ hg ci -Aqm2 -u Bob
52
52
53 $ hg log -r "extra('branch', 'a-b-c-')" --template '{rev}\n'
53 $ hg log -r "extra('branch', 'a-b-c-')" --template '{rev}\n'
54 2
54 2
55 $ hg log -r "extra('branch')" --template '{rev}\n'
55 $ hg log -r "extra('branch')" --template '{rev}\n'
56 0
56 0
57 1
57 1
58 2
58 2
59 $ hg log -r "extra('branch', 're:a')" --template '{rev} {branch}\n'
59 $ hg log -r "extra('branch', 're:a')" --template '{rev} {branch}\n'
60 0 a
60 0 a
61 2 a-b-c-
61 2 a-b-c-
62
62
63 $ hg co 1
63 $ hg co 1
64 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
64 1 files updated, 0 files merged, 0 files removed, 0 files unresolved
65 $ hg branch +a+b+c+
65 $ hg branch +a+b+c+
66 marked working directory as branch +a+b+c+
66 marked working directory as branch +a+b+c+
67 $ hg ci -Aqm3
67 $ hg ci -Aqm3
68
68
69 $ hg co 2 # interleave
69 $ hg co 2 # interleave
70 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
70 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
71 $ echo bb > b
71 $ echo bb > b
72 $ hg branch -- -a-b-c-
72 $ hg branch -- -a-b-c-
73 marked working directory as branch -a-b-c-
73 marked working directory as branch -a-b-c-
74 $ hg ci -Aqm4 -d "May 12 2005"
74 $ hg ci -Aqm4 -d "May 12 2005"
75
75
76 $ hg co 3
76 $ hg co 3
77 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
77 2 files updated, 0 files merged, 0 files removed, 0 files unresolved
78 $ hg branch !a/b/c/
78 $ hg branch !a/b/c/
79 marked working directory as branch !a/b/c/
79 marked working directory as branch !a/b/c/
80 $ hg ci -Aqm"5 bug"
80 $ hg ci -Aqm"5 bug"
81
81
82 $ hg merge 4
82 $ hg merge 4
83 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
83 1 files updated, 0 files merged, 1 files removed, 0 files unresolved
84 (branch merge, don't forget to commit)
84 (branch merge, don't forget to commit)
85 $ hg branch _a_b_c_
85 $ hg branch _a_b_c_
86 marked working directory as branch _a_b_c_
86 marked working directory as branch _a_b_c_
87 $ hg ci -Aqm"6 issue619"
87 $ hg ci -Aqm"6 issue619"
88
88
89 $ hg branch .a.b.c.
89 $ hg branch .a.b.c.
90 marked working directory as branch .a.b.c.
90 marked working directory as branch .a.b.c.
91 $ hg ci -Aqm7
91 $ hg ci -Aqm7
92
92
93 $ hg branch all
93 $ hg branch all
94 marked working directory as branch all
94 marked working directory as branch all
95
95
96 $ hg co 4
96 $ hg co 4
97 0 files updated, 0 files merged, 0 files removed, 0 files unresolved
97 0 files updated, 0 files merged, 0 files removed, 0 files unresolved
98 $ hg branch Γ©
98 $ hg branch Γ©
99 marked working directory as branch \xc3\xa9 (esc)
99 marked working directory as branch \xc3\xa9 (esc)
100 $ hg ci -Aqm9
100 $ hg ci -Aqm9
101
101
102 $ hg tag -r6 1.0
102 $ hg tag -r6 1.0
103 $ hg bookmark -r6 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
103 $ hg bookmark -r6 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
104
104
105 $ hg clone --quiet -U -r 7 . ../remote1
105 $ hg clone --quiet -U -r 7 . ../remote1
106 $ hg clone --quiet -U -r 8 . ../remote2
106 $ hg clone --quiet -U -r 8 . ../remote2
107 $ echo "[paths]" >> .hg/hgrc
107 $ echo "[paths]" >> .hg/hgrc
108 $ echo "default = ../remote1" >> .hg/hgrc
108 $ echo "default = ../remote1" >> .hg/hgrc
109
109
110 trivial
110 trivial
111
111
112 $ try 0:1
112 $ try 0:1
113 (range
113 (range
114 ('symbol', '0')
114 ('symbol', '0')
115 ('symbol', '1'))
115 ('symbol', '1'))
116 * set:
116 * set:
117 <spanset+ 0:1>
117 <spanset+ 0:1>
118 0
118 0
119 1
119 1
120 $ try --optimize :
120 $ try --optimize :
121 (rangeall
121 (rangeall
122 None)
122 None)
123 * optimized:
123 * optimized:
124 (range
124 (range
125 ('string', '0')
125 ('string', '0')
126 ('string', 'tip'))
126 ('string', 'tip'))
127 * set:
127 * set:
128 <spanset+ 0:9>
128 <spanset+ 0:9>
129 0
129 0
130 1
130 1
131 2
131 2
132 3
132 3
133 4
133 4
134 5
134 5
135 6
135 6
136 7
136 7
137 8
137 8
138 9
138 9
139 $ try 3::6
139 $ try 3::6
140 (dagrange
140 (dagrange
141 ('symbol', '3')
141 ('symbol', '3')
142 ('symbol', '6'))
142 ('symbol', '6'))
143 * set:
143 * set:
144 <baseset+ [3, 5, 6]>
144 <baseset+ [3, 5, 6]>
145 3
145 3
146 5
146 5
147 6
147 6
148 $ try '0|1|2'
148 $ try '0|1|2'
149 (or
149 (or
150 ('symbol', '0')
150 ('symbol', '0')
151 ('symbol', '1')
151 ('symbol', '1')
152 ('symbol', '2'))
152 ('symbol', '2'))
153 * set:
153 * set:
154 <baseset [0, 1, 2]>
154 <baseset [0, 1, 2]>
155 0
155 0
156 1
156 1
157 2
157 2
158
158
159 names that should work without quoting
159 names that should work without quoting
160
160
161 $ try a
161 $ try a
162 ('symbol', 'a')
162 ('symbol', 'a')
163 * set:
163 * set:
164 <baseset [0]>
164 <baseset [0]>
165 0
165 0
166 $ try b-a
166 $ try b-a
167 (minus
167 (minus
168 ('symbol', 'b')
168 ('symbol', 'b')
169 ('symbol', 'a'))
169 ('symbol', 'a'))
170 * set:
170 * set:
171 <filteredset
171 <filteredset
172 <baseset [1]>,
172 <baseset [1]>,
173 <not
173 <not
174 <baseset [0]>>>
174 <baseset [0]>>>
175 1
175 1
176 $ try _a_b_c_
176 $ try _a_b_c_
177 ('symbol', '_a_b_c_')
177 ('symbol', '_a_b_c_')
178 * set:
178 * set:
179 <baseset [6]>
179 <baseset [6]>
180 6
180 6
181 $ try _a_b_c_-a
181 $ try _a_b_c_-a
182 (minus
182 (minus
183 ('symbol', '_a_b_c_')
183 ('symbol', '_a_b_c_')
184 ('symbol', 'a'))
184 ('symbol', 'a'))
185 * set:
185 * set:
186 <filteredset
186 <filteredset
187 <baseset [6]>,
187 <baseset [6]>,
188 <not
188 <not
189 <baseset [0]>>>
189 <baseset [0]>>>
190 6
190 6
191 $ try .a.b.c.
191 $ try .a.b.c.
192 ('symbol', '.a.b.c.')
192 ('symbol', '.a.b.c.')
193 * set:
193 * set:
194 <baseset [7]>
194 <baseset [7]>
195 7
195 7
196 $ try .a.b.c.-a
196 $ try .a.b.c.-a
197 (minus
197 (minus
198 ('symbol', '.a.b.c.')
198 ('symbol', '.a.b.c.')
199 ('symbol', 'a'))
199 ('symbol', 'a'))
200 * set:
200 * set:
201 <filteredset
201 <filteredset
202 <baseset [7]>,
202 <baseset [7]>,
203 <not
203 <not
204 <baseset [0]>>>
204 <baseset [0]>>>
205 7
205 7
206
206
207 names that should be caught by fallback mechanism
207 names that should be caught by fallback mechanism
208
208
209 $ try -- '-a-b-c-'
209 $ try -- '-a-b-c-'
210 ('symbol', '-a-b-c-')
210 ('symbol', '-a-b-c-')
211 * set:
211 * set:
212 <baseset [4]>
212 <baseset [4]>
213 4
213 4
214 $ log -a-b-c-
214 $ log -a-b-c-
215 4
215 4
216 $ try '+a+b+c+'
216 $ try '+a+b+c+'
217 ('symbol', '+a+b+c+')
217 ('symbol', '+a+b+c+')
218 * set:
218 * set:
219 <baseset [3]>
219 <baseset [3]>
220 3
220 3
221 $ try '+a+b+c+:'
221 $ try '+a+b+c+:'
222 (rangepost
222 (rangepost
223 ('symbol', '+a+b+c+'))
223 ('symbol', '+a+b+c+'))
224 * set:
224 * set:
225 <spanset+ 3:9>
225 <spanset+ 3:9>
226 3
226 3
227 4
227 4
228 5
228 5
229 6
229 6
230 7
230 7
231 8
231 8
232 9
232 9
233 $ try ':+a+b+c+'
233 $ try ':+a+b+c+'
234 (rangepre
234 (rangepre
235 ('symbol', '+a+b+c+'))
235 ('symbol', '+a+b+c+'))
236 * set:
236 * set:
237 <spanset+ 0:3>
237 <spanset+ 0:3>
238 0
238 0
239 1
239 1
240 2
240 2
241 3
241 3
242 $ try -- '-a-b-c-:+a+b+c+'
242 $ try -- '-a-b-c-:+a+b+c+'
243 (range
243 (range
244 ('symbol', '-a-b-c-')
244 ('symbol', '-a-b-c-')
245 ('symbol', '+a+b+c+'))
245 ('symbol', '+a+b+c+'))
246 * set:
246 * set:
247 <spanset- 3:4>
247 <spanset- 3:4>
248 4
248 4
249 3
249 3
250 $ log '-a-b-c-:+a+b+c+'
250 $ log '-a-b-c-:+a+b+c+'
251 4
251 4
252 3
252 3
253
253
254 $ try -- -a-b-c--a # complains
254 $ try -- -a-b-c--a # complains
255 (minus
255 (minus
256 (minus
256 (minus
257 (minus
257 (minus
258 (negate
258 (negate
259 ('symbol', 'a'))
259 ('symbol', 'a'))
260 ('symbol', 'b'))
260 ('symbol', 'b'))
261 ('symbol', 'c'))
261 ('symbol', 'c'))
262 (negate
262 (negate
263 ('symbol', 'a')))
263 ('symbol', 'a')))
264 abort: unknown revision '-a'!
264 abort: unknown revision '-a'!
265 [255]
265 [255]
266 $ try Γ©
266 $ try Γ©
267 ('symbol', '\xc3\xa9')
267 ('symbol', '\xc3\xa9')
268 * set:
268 * set:
269 <baseset [9]>
269 <baseset [9]>
270 9
270 9
271
271
272 no quoting needed
272 no quoting needed
273
273
274 $ log ::a-b-c-
274 $ log ::a-b-c-
275 0
275 0
276 1
276 1
277 2
277 2
278
278
279 quoting needed
279 quoting needed
280
280
281 $ try '"-a-b-c-"-a'
281 $ try '"-a-b-c-"-a'
282 (minus
282 (minus
283 ('string', '-a-b-c-')
283 ('string', '-a-b-c-')
284 ('symbol', 'a'))
284 ('symbol', 'a'))
285 * set:
285 * set:
286 <filteredset
286 <filteredset
287 <baseset [4]>,
287 <baseset [4]>,
288 <not
288 <not
289 <baseset [0]>>>
289 <baseset [0]>>>
290 4
290 4
291
291
292 $ log '1 or 2'
292 $ log '1 or 2'
293 1
293 1
294 2
294 2
295 $ log '1|2'
295 $ log '1|2'
296 1
296 1
297 2
297 2
298 $ log '1 and 2'
298 $ log '1 and 2'
299 $ log '1&2'
299 $ log '1&2'
300 $ try '1&2|3' # precedence - and is higher
300 $ try '1&2|3' # precedence - and is higher
301 (or
301 (or
302 (and
302 (and
303 ('symbol', '1')
303 ('symbol', '1')
304 ('symbol', '2'))
304 ('symbol', '2'))
305 ('symbol', '3'))
305 ('symbol', '3'))
306 * set:
306 * set:
307 <addset
307 <addset
308 <baseset []>,
308 <baseset []>,
309 <baseset [3]>>
309 <baseset [3]>>
310 3
310 3
311 $ try '1|2&3'
311 $ try '1|2&3'
312 (or
312 (or
313 ('symbol', '1')
313 ('symbol', '1')
314 (and
314 (and
315 ('symbol', '2')
315 ('symbol', '2')
316 ('symbol', '3')))
316 ('symbol', '3')))
317 * set:
317 * set:
318 <addset
318 <addset
319 <baseset [1]>,
319 <baseset [1]>,
320 <baseset []>>
320 <baseset []>>
321 1
321 1
322 $ try '1&2&3' # associativity
322 $ try '1&2&3' # associativity
323 (and
323 (and
324 (and
324 (and
325 ('symbol', '1')
325 ('symbol', '1')
326 ('symbol', '2'))
326 ('symbol', '2'))
327 ('symbol', '3'))
327 ('symbol', '3'))
328 * set:
328 * set:
329 <baseset []>
329 <baseset []>
330 $ try '1|(2|3)'
330 $ try '1|(2|3)'
331 (or
331 (or
332 ('symbol', '1')
332 ('symbol', '1')
333 (group
333 (group
334 (or
334 (or
335 ('symbol', '2')
335 ('symbol', '2')
336 ('symbol', '3'))))
336 ('symbol', '3'))))
337 * set:
337 * set:
338 <addset
338 <addset
339 <baseset [1]>,
339 <baseset [1]>,
340 <baseset [2, 3]>>
340 <baseset [2, 3]>>
341 1
341 1
342 2
342 2
343 3
343 3
344 $ log '1.0' # tag
344 $ log '1.0' # tag
345 6
345 6
346 $ log 'a' # branch
346 $ log 'a' # branch
347 0
347 0
348 $ log '2785f51ee'
348 $ log '2785f51ee'
349 0
349 0
350 $ log 'date(2005)'
350 $ log 'date(2005)'
351 4
351 4
352 $ log 'date(this is a test)'
352 $ log 'date(this is a test)'
353 hg: parse error at 10: unexpected token: symbol
353 hg: parse error at 10: unexpected token: symbol
354 [255]
354 [255]
355 $ log 'date()'
355 $ log 'date()'
356 hg: parse error: date requires a string
356 hg: parse error: date requires a string
357 [255]
357 [255]
358 $ log 'date'
358 $ log 'date'
359 abort: unknown revision 'date'!
359 abort: unknown revision 'date'!
360 [255]
360 [255]
361 $ log 'date('
361 $ log 'date('
362 hg: parse error at 5: not a prefix: end
362 hg: parse error at 5: not a prefix: end
363 [255]
363 [255]
364 $ log 'date("\xy")'
364 $ log 'date("\xy")'
365 hg: parse error: invalid \x escape
365 hg: parse error: invalid \x escape
366 [255]
366 [255]
367 $ log 'date(tip)'
367 $ log 'date(tip)'
368 abort: invalid date: 'tip'
368 abort: invalid date: 'tip'
369 [255]
369 [255]
370 $ log '0:date'
370 $ log '0:date'
371 abort: unknown revision 'date'!
371 abort: unknown revision 'date'!
372 [255]
372 [255]
373 $ log '::"date"'
373 $ log '::"date"'
374 abort: unknown revision 'date'!
374 abort: unknown revision 'date'!
375 [255]
375 [255]
376 $ hg book date -r 4
376 $ hg book date -r 4
377 $ log '0:date'
377 $ log '0:date'
378 0
378 0
379 1
379 1
380 2
380 2
381 3
381 3
382 4
382 4
383 $ log '::date'
383 $ log '::date'
384 0
384 0
385 1
385 1
386 2
386 2
387 4
387 4
388 $ log '::"date"'
388 $ log '::"date"'
389 0
389 0
390 1
390 1
391 2
391 2
392 4
392 4
393 $ log 'date(2005) and 1::'
393 $ log 'date(2005) and 1::'
394 4
394 4
395 $ hg book -d date
395 $ hg book -d date
396
396
397 keyword arguments
397 keyword arguments
398
398
399 $ log 'extra(branch, value=a)'
399 $ log 'extra(branch, value=a)'
400 0
400 0
401
401
402 $ log 'extra(branch, a, b)'
402 $ log 'extra(branch, a, b)'
403 hg: parse error: extra takes at most 2 arguments
403 hg: parse error: extra takes at most 2 arguments
404 [255]
404 [255]
405 $ log 'extra(a, label=b)'
405 $ log 'extra(a, label=b)'
406 hg: parse error: extra got multiple values for keyword argument 'label'
406 hg: parse error: extra got multiple values for keyword argument 'label'
407 [255]
407 [255]
408 $ log 'extra(label=branch, default)'
408 $ log 'extra(label=branch, default)'
409 hg: parse error: extra got an invalid argument
409 hg: parse error: extra got an invalid argument
410 [255]
410 [255]
411 $ log 'extra(branch, foo+bar=baz)'
411 $ log 'extra(branch, foo+bar=baz)'
412 hg: parse error: extra got an invalid argument
412 hg: parse error: extra got an invalid argument
413 [255]
413 [255]
414 $ log 'extra(unknown=branch)'
414 $ log 'extra(unknown=branch)'
415 hg: parse error: extra got an unexpected keyword argument 'unknown'
415 hg: parse error: extra got an unexpected keyword argument 'unknown'
416 [255]
416 [255]
417
417
418 $ try 'foo=bar|baz'
418 $ try 'foo=bar|baz'
419 (keyvalue
419 (keyvalue
420 ('symbol', 'foo')
420 ('symbol', 'foo')
421 (or
421 (or
422 ('symbol', 'bar')
422 ('symbol', 'bar')
423 ('symbol', 'baz')))
423 ('symbol', 'baz')))
424 hg: parse error: can't use a key-value pair in this context
424 hg: parse error: can't use a key-value pair in this context
425 [255]
425 [255]
426
426
427 Test that symbols only get parsed as functions if there's an opening
427 Test that symbols only get parsed as functions if there's an opening
428 parenthesis.
428 parenthesis.
429
429
430 $ hg book only -r 9
430 $ hg book only -r 9
431 $ log 'only(only)' # Outer "only" is a function, inner "only" is the bookmark
431 $ log 'only(only)' # Outer "only" is a function, inner "only" is the bookmark
432 8
432 8
433 9
433 9
434
434
435 ancestor can accept 0 or more arguments
435 ancestor can accept 0 or more arguments
436
436
437 $ log 'ancestor()'
437 $ log 'ancestor()'
438 $ log 'ancestor(1)'
438 $ log 'ancestor(1)'
439 1
439 1
440 $ log 'ancestor(4,5)'
440 $ log 'ancestor(4,5)'
441 1
441 1
442 $ log 'ancestor(4,5) and 4'
442 $ log 'ancestor(4,5) and 4'
443 $ log 'ancestor(0,0,1,3)'
443 $ log 'ancestor(0,0,1,3)'
444 0
444 0
445 $ log 'ancestor(3,1,5,3,5,1)'
445 $ log 'ancestor(3,1,5,3,5,1)'
446 1
446 1
447 $ log 'ancestor(0,1,3,5)'
447 $ log 'ancestor(0,1,3,5)'
448 0
448 0
449 $ log 'ancestor(1,2,3,4,5)'
449 $ log 'ancestor(1,2,3,4,5)'
450 1
450 1
451
451
452 test ancestors
452 test ancestors
453
453
454 $ log 'ancestors(5)'
454 $ log 'ancestors(5)'
455 0
455 0
456 1
456 1
457 3
457 3
458 5
458 5
459 $ log 'ancestor(ancestors(5))'
459 $ log 'ancestor(ancestors(5))'
460 0
460 0
461 $ log '::r3232()'
461 $ log '::r3232()'
462 0
462 0
463 1
463 1
464 2
464 2
465 3
465 3
466
466
467 $ log 'author(bob)'
467 $ log 'author(bob)'
468 2
468 2
469 $ log 'author("re:bob|test")'
469 $ log 'author("re:bob|test")'
470 0
470 0
471 1
471 1
472 2
472 2
473 3
473 3
474 4
474 4
475 5
475 5
476 6
476 6
477 7
477 7
478 8
478 8
479 9
479 9
480 $ log 'branch(Γ©)'
480 $ log 'branch(Γ©)'
481 8
481 8
482 9
482 9
483 $ log 'branch(a)'
483 $ log 'branch(a)'
484 0
484 0
485 $ hg log -r 'branch("re:a")' --template '{rev} {branch}\n'
485 $ hg log -r 'branch("re:a")' --template '{rev} {branch}\n'
486 0 a
486 0 a
487 2 a-b-c-
487 2 a-b-c-
488 3 +a+b+c+
488 3 +a+b+c+
489 4 -a-b-c-
489 4 -a-b-c-
490 5 !a/b/c/
490 5 !a/b/c/
491 6 _a_b_c_
491 6 _a_b_c_
492 7 .a.b.c.
492 7 .a.b.c.
493 $ log 'children(ancestor(4,5))'
493 $ log 'children(ancestor(4,5))'
494 2
494 2
495 3
495 3
496 $ log 'closed()'
496 $ log 'closed()'
497 $ log 'contains(a)'
497 $ log 'contains(a)'
498 0
498 0
499 1
499 1
500 3
500 3
501 5
501 5
502 $ log 'contains("../repo/a")'
502 $ log 'contains("../repo/a")'
503 0
503 0
504 1
504 1
505 3
505 3
506 5
506 5
507 $ log 'desc(B)'
507 $ log 'desc(B)'
508 5
508 5
509 $ log 'descendants(2 or 3)'
509 $ log 'descendants(2 or 3)'
510 2
510 2
511 3
511 3
512 4
512 4
513 5
513 5
514 6
514 6
515 7
515 7
516 8
516 8
517 9
517 9
518 $ log 'file("b*")'
518 $ log 'file("b*")'
519 1
519 1
520 4
520 4
521 $ log 'filelog("b")'
521 $ log 'filelog("b")'
522 1
522 1
523 4
523 4
524 $ log 'filelog("../repo/b")'
524 $ log 'filelog("../repo/b")'
525 1
525 1
526 4
526 4
527 $ log 'follow()'
527 $ log 'follow()'
528 0
528 0
529 1
529 1
530 2
530 2
531 4
531 4
532 8
532 8
533 9
533 9
534 $ log 'grep("issue\d+")'
534 $ log 'grep("issue\d+")'
535 6
535 6
536 $ try 'grep("(")' # invalid regular expression
536 $ try 'grep("(")' # invalid regular expression
537 (func
537 (func
538 ('symbol', 'grep')
538 ('symbol', 'grep')
539 ('string', '('))
539 ('string', '('))
540 hg: parse error: invalid match pattern: unbalanced parenthesis
540 hg: parse error: invalid match pattern: unbalanced parenthesis
541 [255]
541 [255]
542 $ try 'grep("\bissue\d+")'
542 $ try 'grep("\bissue\d+")'
543 (func
543 (func
544 ('symbol', 'grep')
544 ('symbol', 'grep')
545 ('string', '\x08issue\\d+'))
545 ('string', '\x08issue\\d+'))
546 * set:
546 * set:
547 <filteredset
547 <filteredset
548 <fullreposet+ 0:9>,
548 <fullreposet+ 0:9>,
549 <grep '\x08issue\\d+'>>
549 <grep '\x08issue\\d+'>>
550 $ try 'grep(r"\bissue\d+")'
550 $ try 'grep(r"\bissue\d+")'
551 (func
551 (func
552 ('symbol', 'grep')
552 ('symbol', 'grep')
553 ('string', '\\bissue\\d+'))
553 ('string', '\\bissue\\d+'))
554 * set:
554 * set:
555 <filteredset
555 <filteredset
556 <fullreposet+ 0:9>,
556 <fullreposet+ 0:9>,
557 <grep '\\bissue\\d+'>>
557 <grep '\\bissue\\d+'>>
558 6
558 6
559 $ try 'grep(r"\")'
559 $ try 'grep(r"\")'
560 hg: parse error at 7: unterminated string
560 hg: parse error at 7: unterminated string
561 [255]
561 [255]
562 $ log 'head()'
562 $ log 'head()'
563 0
563 0
564 1
564 1
565 2
565 2
566 3
566 3
567 4
567 4
568 5
568 5
569 6
569 6
570 7
570 7
571 9
571 9
572 $ log 'heads(6::)'
572 $ log 'heads(6::)'
573 7
573 7
574 $ log 'keyword(issue)'
574 $ log 'keyword(issue)'
575 6
575 6
576 $ log 'keyword("test a")'
576 $ log 'keyword("test a")'
577 $ log 'limit(head(), 1)'
577 $ log 'limit(head(), 1)'
578 0
578 0
579 $ log 'limit(author("re:bob|test"), 3, 5)'
579 $ log 'limit(author("re:bob|test"), 3, 5)'
580 5
580 5
581 6
581 6
582 7
582 7
583 $ log 'limit(author("re:bob|test"), offset=6)'
583 $ log 'limit(author("re:bob|test"), offset=6)'
584 6
584 6
585 $ log 'limit(author("re:bob|test"), offset=10)'
585 $ log 'limit(author("re:bob|test"), offset=10)'
586 $ log 'limit(all(), 1, -1)'
586 $ log 'limit(all(), 1, -1)'
587 hg: parse error: negative offset
587 hg: parse error: negative offset
588 [255]
588 [255]
589 $ log 'matching(6)'
589 $ log 'matching(6)'
590 6
590 6
591 $ log 'matching(6:7, "phase parents user date branch summary files description substate")'
591 $ log 'matching(6:7, "phase parents user date branch summary files description substate")'
592 6
592 6
593 7
593 7
594
594
595 Testing min and max
595 Testing min and max
596
596
597 max: simple
597 max: simple
598
598
599 $ log 'max(contains(a))'
599 $ log 'max(contains(a))'
600 5
600 5
601
601
602 max: simple on unordered set)
602 max: simple on unordered set)
603
603
604 $ log 'max((4+0+2+5+7) and contains(a))'
604 $ log 'max((4+0+2+5+7) and contains(a))'
605 5
605 5
606
606
607 max: no result
607 max: no result
608
608
609 $ log 'max(contains(stringthatdoesnotappearanywhere))'
609 $ log 'max(contains(stringthatdoesnotappearanywhere))'
610
610
611 max: no result on unordered set
611 max: no result on unordered set
612
612
613 $ log 'max((4+0+2+5+7) and contains(stringthatdoesnotappearanywhere))'
613 $ log 'max((4+0+2+5+7) and contains(stringthatdoesnotappearanywhere))'
614
614
615 min: simple
615 min: simple
616
616
617 $ log 'min(contains(a))'
617 $ log 'min(contains(a))'
618 0
618 0
619
619
620 min: simple on unordered set
620 min: simple on unordered set
621
621
622 $ log 'min((4+0+2+5+7) and contains(a))'
622 $ log 'min((4+0+2+5+7) and contains(a))'
623 0
623 0
624
624
625 min: empty
625 min: empty
626
626
627 $ log 'min(contains(stringthatdoesnotappearanywhere))'
627 $ log 'min(contains(stringthatdoesnotappearanywhere))'
628
628
629 min: empty on unordered set
629 min: empty on unordered set
630
630
631 $ log 'min((4+0+2+5+7) and contains(stringthatdoesnotappearanywhere))'
631 $ log 'min((4+0+2+5+7) and contains(stringthatdoesnotappearanywhere))'
632
632
633
633
634 $ log 'merge()'
634 $ log 'merge()'
635 6
635 6
636 $ log 'branchpoint()'
636 $ log 'branchpoint()'
637 1
637 1
638 4
638 4
639 $ log 'modifies(b)'
639 $ log 'modifies(b)'
640 4
640 4
641 $ log 'modifies("path:b")'
641 $ log 'modifies("path:b")'
642 4
642 4
643 $ log 'modifies("*")'
643 $ log 'modifies("*")'
644 4
644 4
645 6
645 6
646 $ log 'modifies("set:modified()")'
646 $ log 'modifies("set:modified()")'
647 4
647 4
648 $ log 'id(5)'
648 $ log 'id(5)'
649 2
649 2
650 $ log 'only(9)'
650 $ log 'only(9)'
651 8
651 8
652 9
652 9
653 $ log 'only(8)'
653 $ log 'only(8)'
654 8
654 8
655 $ log 'only(9, 5)'
655 $ log 'only(9, 5)'
656 2
656 2
657 4
657 4
658 8
658 8
659 9
659 9
660 $ log 'only(7 + 9, 5 + 2)'
660 $ log 'only(7 + 9, 5 + 2)'
661 4
661 4
662 6
662 6
663 7
663 7
664 8
664 8
665 9
665 9
666
666
667 Test empty set input
667 Test empty set input
668 $ log 'only(p2())'
668 $ log 'only(p2())'
669 $ log 'only(p1(), p2())'
669 $ log 'only(p1(), p2())'
670 0
670 0
671 1
671 1
672 2
672 2
673 4
673 4
674 8
674 8
675 9
675 9
676
676
677 Test '%' operator
677 Test '%' operator
678
678
679 $ log '9%'
679 $ log '9%'
680 8
680 8
681 9
681 9
682 $ log '9%5'
682 $ log '9%5'
683 2
683 2
684 4
684 4
685 8
685 8
686 9
686 9
687 $ log '(7 + 9)%(5 + 2)'
687 $ log '(7 + 9)%(5 + 2)'
688 4
688 4
689 6
689 6
690 7
690 7
691 8
691 8
692 9
692 9
693
693
694 Test opreand of '%' is optimized recursively (issue4670)
694 Test opreand of '%' is optimized recursively (issue4670)
695
695
696 $ try --optimize '8:9-8%'
696 $ try --optimize '8:9-8%'
697 (onlypost
697 (onlypost
698 (minus
698 (minus
699 (range
699 (range
700 ('symbol', '8')
700 ('symbol', '8')
701 ('symbol', '9'))
701 ('symbol', '9'))
702 ('symbol', '8')))
702 ('symbol', '8')))
703 * optimized:
703 * optimized:
704 (func
704 (func
705 ('symbol', 'only')
705 ('symbol', 'only')
706 (difference
706 (difference
707 (range
707 (range
708 ('symbol', '8')
708 ('symbol', '8')
709 ('symbol', '9'))
709 ('symbol', '9'))
710 ('symbol', '8')))
710 ('symbol', '8')))
711 * set:
711 * set:
712 <baseset+ [8, 9]>
712 <baseset+ [8, 9]>
713 8
713 8
714 9
714 9
715 $ try --optimize '(9)%(5)'
715 $ try --optimize '(9)%(5)'
716 (only
716 (only
717 (group
717 (group
718 ('symbol', '9'))
718 ('symbol', '9'))
719 (group
719 (group
720 ('symbol', '5')))
720 ('symbol', '5')))
721 * optimized:
721 * optimized:
722 (func
722 (func
723 ('symbol', 'only')
723 ('symbol', 'only')
724 (list
724 (list
725 ('symbol', '9')
725 ('symbol', '9')
726 ('symbol', '5')))
726 ('symbol', '5')))
727 * set:
727 * set:
728 <baseset+ [2, 4, 8, 9]>
728 <baseset+ [2, 4, 8, 9]>
729 2
729 2
730 4
730 4
731 8
731 8
732 9
732 9
733
733
734 Test the order of operations
734 Test the order of operations
735
735
736 $ log '7 + 9%5 + 2'
736 $ log '7 + 9%5 + 2'
737 7
737 7
738 2
738 2
739 4
739 4
740 8
740 8
741 9
741 9
742
742
743 Test explicit numeric revision
743 Test explicit numeric revision
744 $ log 'rev(-2)'
744 $ log 'rev(-2)'
745 $ log 'rev(-1)'
745 $ log 'rev(-1)'
746 -1
746 -1
747 $ log 'rev(0)'
747 $ log 'rev(0)'
748 0
748 0
749 $ log 'rev(9)'
749 $ log 'rev(9)'
750 9
750 9
751 $ log 'rev(10)'
751 $ log 'rev(10)'
752 $ log 'rev(tip)'
752 $ log 'rev(tip)'
753 hg: parse error: rev expects a number
753 hg: parse error: rev expects a number
754 [255]
754 [255]
755
755
756 Test hexadecimal revision
756 Test hexadecimal revision
757 $ log 'id(2)'
757 $ log 'id(2)'
758 abort: 00changelog.i@2: ambiguous identifier!
758 abort: 00changelog.i@2: ambiguous identifier!
759 [255]
759 [255]
760 $ log 'id(23268)'
760 $ log 'id(23268)'
761 4
761 4
762 $ log 'id(2785f51eece)'
762 $ log 'id(2785f51eece)'
763 0
763 0
764 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532c)'
764 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532c)'
765 8
765 8
766 $ log 'id(d5d0dcbdc4a)'
766 $ log 'id(d5d0dcbdc4a)'
767 $ log 'id(d5d0dcbdc4w)'
767 $ log 'id(d5d0dcbdc4w)'
768 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532d)'
768 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532d)'
769 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532q)'
769 $ log 'id(d5d0dcbdc4d9ff5dbb2d336f32f0bb561c1a532q)'
770 $ log 'id(1.0)'
770 $ log 'id(1.0)'
771 $ log 'id(xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx)'
771 $ log 'id(xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx)'
772
772
773 Test null revision
773 Test null revision
774 $ log '(null)'
774 $ log '(null)'
775 -1
775 -1
776 $ log '(null:0)'
776 $ log '(null:0)'
777 -1
777 -1
778 0
778 0
779 $ log '(0:null)'
779 $ log '(0:null)'
780 0
780 0
781 -1
781 -1
782 $ log 'null::0'
782 $ log 'null::0'
783 -1
783 -1
784 0
784 0
785 $ log 'null:tip - 0:'
785 $ log 'null:tip - 0:'
786 -1
786 -1
787 $ log 'null: and null::' | head -1
787 $ log 'null: and null::' | head -1
788 -1
788 -1
789 $ log 'null: or 0:' | head -2
789 $ log 'null: or 0:' | head -2
790 -1
790 -1
791 0
791 0
792 $ log 'ancestors(null)'
792 $ log 'ancestors(null)'
793 -1
793 -1
794 $ log 'reverse(null:)' | tail -2
794 $ log 'reverse(null:)' | tail -2
795 0
795 0
796 -1
796 -1
797 BROKEN: should be '-1'
797 BROKEN: should be '-1'
798 $ log 'first(null:)'
798 $ log 'first(null:)'
799 BROKEN: should be '-1'
799 BROKEN: should be '-1'
800 $ log 'min(null:)'
800 $ log 'min(null:)'
801 $ log 'tip:null and all()' | tail -2
801 $ log 'tip:null and all()' | tail -2
802 1
802 1
803 0
803 0
804
804
805 Test working-directory revision
805 Test working-directory revision
806 $ hg debugrevspec 'wdir()'
806 $ hg debugrevspec 'wdir()'
807 2147483647
807 2147483647
808 $ hg debugrevspec 'tip or wdir()'
808 $ hg debugrevspec 'tip or wdir()'
809 9
809 9
810 2147483647
810 2147483647
811 $ hg debugrevspec '0:tip and wdir()'
811 $ hg debugrevspec '0:tip and wdir()'
812 $ log '0:wdir()' | tail -3
812 $ log '0:wdir()' | tail -3
813 8
813 8
814 9
814 9
815 2147483647
815 2147483647
816 $ log 'wdir():0' | head -3
816 $ log 'wdir():0' | head -3
817 2147483647
817 2147483647
818 9
818 9
819 8
819 8
820 $ log 'wdir():wdir()'
820 $ log 'wdir():wdir()'
821 2147483647
821 2147483647
822 $ log '(all() + wdir()) & min(. + wdir())'
822 $ log '(all() + wdir()) & min(. + wdir())'
823 9
823 9
824 $ log '(all() + wdir()) & max(. + wdir())'
824 $ log '(all() + wdir()) & max(. + wdir())'
825 2147483647
825 2147483647
826 $ log '(all() + wdir()) & first(wdir() + .)'
826 $ log '(all() + wdir()) & first(wdir() + .)'
827 2147483647
827 2147483647
828 $ log '(all() + wdir()) & last(. + wdir())'
828 $ log '(all() + wdir()) & last(. + wdir())'
829 2147483647
829 2147483647
830
830
831 $ log 'outgoing()'
831 $ log 'outgoing()'
832 8
832 8
833 9
833 9
834 $ log 'outgoing("../remote1")'
834 $ log 'outgoing("../remote1")'
835 8
835 8
836 9
836 9
837 $ log 'outgoing("../remote2")'
837 $ log 'outgoing("../remote2")'
838 3
838 3
839 5
839 5
840 6
840 6
841 7
841 7
842 9
842 9
843 $ log 'p1(merge())'
843 $ log 'p1(merge())'
844 5
844 5
845 $ log 'p2(merge())'
845 $ log 'p2(merge())'
846 4
846 4
847 $ log 'parents(merge())'
847 $ log 'parents(merge())'
848 4
848 4
849 5
849 5
850 $ log 'p1(branchpoint())'
850 $ log 'p1(branchpoint())'
851 0
851 0
852 2
852 2
853 $ log 'p2(branchpoint())'
853 $ log 'p2(branchpoint())'
854 $ log 'parents(branchpoint())'
854 $ log 'parents(branchpoint())'
855 0
855 0
856 2
856 2
857 $ log 'removes(a)'
857 $ log 'removes(a)'
858 2
858 2
859 6
859 6
860 $ log 'roots(all())'
860 $ log 'roots(all())'
861 0
861 0
862 $ log 'reverse(2 or 3 or 4 or 5)'
862 $ log 'reverse(2 or 3 or 4 or 5)'
863 5
863 5
864 4
864 4
865 3
865 3
866 2
866 2
867 $ log 'reverse(all())'
867 $ log 'reverse(all())'
868 9
868 9
869 8
869 8
870 7
870 7
871 6
871 6
872 5
872 5
873 4
873 4
874 3
874 3
875 2
875 2
876 1
876 1
877 0
877 0
878 $ log 'reverse(all()) & filelog(b)'
878 $ log 'reverse(all()) & filelog(b)'
879 4
879 4
880 1
880 1
881 $ log 'rev(5)'
881 $ log 'rev(5)'
882 5
882 5
883 $ log 'sort(limit(reverse(all()), 3))'
883 $ log 'sort(limit(reverse(all()), 3))'
884 7
884 7
885 8
885 8
886 9
886 9
887 $ log 'sort(2 or 3 or 4 or 5, date)'
887 $ log 'sort(2 or 3 or 4 or 5, date)'
888 2
888 2
889 3
889 3
890 5
890 5
891 4
891 4
892 $ log 'tagged()'
892 $ log 'tagged()'
893 6
893 6
894 $ log 'tag()'
894 $ log 'tag()'
895 6
895 6
896 $ log 'tag(1.0)'
896 $ log 'tag(1.0)'
897 6
897 6
898 $ log 'tag(tip)'
898 $ log 'tag(tip)'
899 9
899 9
900
900
901 Test order of revisions in compound expression
901 Test order of revisions in compound expression
902 ----------------------------------------------
902 ----------------------------------------------
903
903
904 'A & B' should follow the order of 'A':
904 'A & B' should follow the order of 'A':
905
905
906 $ log '2:0 & 0::2'
906 $ log '2:0 & 0::2'
907 2
907 2
908 1
908 1
909 0
909 0
910
910
911 test sort revset
911 test sort revset
912 --------------------------------------------
912 --------------------------------------------
913
913
914 test when adding two unordered revsets
914 test when adding two unordered revsets
915
915
916 $ log 'sort(keyword(issue) or modifies(b))'
916 $ log 'sort(keyword(issue) or modifies(b))'
917 4
917 4
918 6
918 6
919
919
920 test when sorting a reversed collection in the same way it is
920 test when sorting a reversed collection in the same way it is
921
921
922 $ log 'sort(reverse(all()), -rev)'
922 $ log 'sort(reverse(all()), -rev)'
923 9
923 9
924 8
924 8
925 7
925 7
926 6
926 6
927 5
927 5
928 4
928 4
929 3
929 3
930 2
930 2
931 1
931 1
932 0
932 0
933
933
934 test when sorting a reversed collection
934 test when sorting a reversed collection
935
935
936 $ log 'sort(reverse(all()), rev)'
936 $ log 'sort(reverse(all()), rev)'
937 0
937 0
938 1
938 1
939 2
939 2
940 3
940 3
941 4
941 4
942 5
942 5
943 6
943 6
944 7
944 7
945 8
945 8
946 9
946 9
947
947
948
948
949 test sorting two sorted collections in different orders
949 test sorting two sorted collections in different orders
950
950
951 $ log 'sort(outgoing() or reverse(removes(a)), rev)'
951 $ log 'sort(outgoing() or reverse(removes(a)), rev)'
952 2
952 2
953 6
953 6
954 8
954 8
955 9
955 9
956
956
957 test sorting two sorted collections in different orders backwards
957 test sorting two sorted collections in different orders backwards
958
958
959 $ log 'sort(outgoing() or reverse(removes(a)), -rev)'
959 $ log 'sort(outgoing() or reverse(removes(a)), -rev)'
960 9
960 9
961 8
961 8
962 6
962 6
963 2
963 2
964
964
965 test empty sort key which is noop
966
967 $ log 'sort(0 + 2 + 1, "")'
968 0
969 2
970 1
971
965 test invalid sort keys
972 test invalid sort keys
966
973
967 $ log 'sort(all(), -invalid)'
974 $ log 'sort(all(), -invalid)'
968 hg: parse error: unknown sort key '-invalid'
975 hg: parse error: unknown sort key '-invalid'
969 [255]
976 [255]
970
977
971 $ cd ..
978 $ cd ..
972
979
973 test sorting by multiple keys including variable-length strings
980 test sorting by multiple keys including variable-length strings
974
981
975 $ hg init sorting
982 $ hg init sorting
976 $ cd sorting
983 $ cd sorting
977 $ cat <<EOF >> .hg/hgrc
984 $ cat <<EOF >> .hg/hgrc
978 > [ui]
985 > [ui]
979 > logtemplate = '{rev} {branch|p5}{desc|p5}{author|p5}{date|hgdate}\n'
986 > logtemplate = '{rev} {branch|p5}{desc|p5}{author|p5}{date|hgdate}\n'
980 > [templatealias]
987 > [templatealias]
981 > p5(s) = pad(s, 5)
988 > p5(s) = pad(s, 5)
982 > EOF
989 > EOF
983 $ hg branch -qf b12
990 $ hg branch -qf b12
984 $ hg ci -m m111 -u u112 -d '111 10800'
991 $ hg ci -m m111 -u u112 -d '111 10800'
985 $ hg branch -qf b11
992 $ hg branch -qf b11
986 $ hg ci -m m12 -u u111 -d '112 7200'
993 $ hg ci -m m12 -u u111 -d '112 7200'
987 $ hg branch -qf b111
994 $ hg branch -qf b111
988 $ hg ci -m m11 -u u12 -d '111 3600'
995 $ hg ci -m m11 -u u12 -d '111 3600'
989 $ hg branch -qf b112
996 $ hg branch -qf b112
990 $ hg ci -m m111 -u u11 -d '120 0'
997 $ hg ci -m m111 -u u11 -d '120 0'
991 $ hg branch -qf b111
998 $ hg branch -qf b111
992 $ hg ci -m m112 -u u111 -d '110 14400'
999 $ hg ci -m m112 -u u111 -d '110 14400'
993 created new head
1000 created new head
994
1001
995 compare revisions (has fast path):
1002 compare revisions (has fast path):
996
1003
997 $ hg log -r 'sort(all(), rev)'
1004 $ hg log -r 'sort(all(), rev)'
998 0 b12 m111 u112 111 10800
1005 0 b12 m111 u112 111 10800
999 1 b11 m12 u111 112 7200
1006 1 b11 m12 u111 112 7200
1000 2 b111 m11 u12 111 3600
1007 2 b111 m11 u12 111 3600
1001 3 b112 m111 u11 120 0
1008 3 b112 m111 u11 120 0
1002 4 b111 m112 u111 110 14400
1009 4 b111 m112 u111 110 14400
1003
1010
1004 $ hg log -r 'sort(all(), -rev)'
1011 $ hg log -r 'sort(all(), -rev)'
1005 4 b111 m112 u111 110 14400
1012 4 b111 m112 u111 110 14400
1006 3 b112 m111 u11 120 0
1013 3 b112 m111 u11 120 0
1007 2 b111 m11 u12 111 3600
1014 2 b111 m11 u12 111 3600
1008 1 b11 m12 u111 112 7200
1015 1 b11 m12 u111 112 7200
1009 0 b12 m111 u112 111 10800
1016 0 b12 m111 u112 111 10800
1010
1017
1011 compare variable-length strings (issue5218):
1018 compare variable-length strings (issue5218):
1012
1019
1013 $ hg log -r 'sort(all(), branch)'
1020 $ hg log -r 'sort(all(), branch)'
1014 1 b11 m12 u111 112 7200
1021 1 b11 m12 u111 112 7200
1015 2 b111 m11 u12 111 3600
1022 2 b111 m11 u12 111 3600
1016 4 b111 m112 u111 110 14400
1023 4 b111 m112 u111 110 14400
1017 3 b112 m111 u11 120 0
1024 3 b112 m111 u11 120 0
1018 0 b12 m111 u112 111 10800
1025 0 b12 m111 u112 111 10800
1019
1026
1020 $ hg log -r 'sort(all(), -branch)'
1027 $ hg log -r 'sort(all(), -branch)'
1021 0 b12 m111 u112 111 10800
1028 0 b12 m111 u112 111 10800
1022 3 b112 m111 u11 120 0
1029 3 b112 m111 u11 120 0
1023 2 b111 m11 u12 111 3600
1030 2 b111 m11 u12 111 3600
1024 4 b111 m112 u111 110 14400
1031 4 b111 m112 u111 110 14400
1025 1 b11 m12 u111 112 7200
1032 1 b11 m12 u111 112 7200
1026
1033
1027 $ hg log -r 'sort(all(), desc)'
1034 $ hg log -r 'sort(all(), desc)'
1028 2 b111 m11 u12 111 3600
1035 2 b111 m11 u12 111 3600
1029 0 b12 m111 u112 111 10800
1036 0 b12 m111 u112 111 10800
1030 3 b112 m111 u11 120 0
1037 3 b112 m111 u11 120 0
1031 4 b111 m112 u111 110 14400
1038 4 b111 m112 u111 110 14400
1032 1 b11 m12 u111 112 7200
1039 1 b11 m12 u111 112 7200
1033
1040
1034 $ hg log -r 'sort(all(), -desc)'
1041 $ hg log -r 'sort(all(), -desc)'
1035 1 b11 m12 u111 112 7200
1042 1 b11 m12 u111 112 7200
1036 4 b111 m112 u111 110 14400
1043 4 b111 m112 u111 110 14400
1037 0 b12 m111 u112 111 10800
1044 0 b12 m111 u112 111 10800
1038 3 b112 m111 u11 120 0
1045 3 b112 m111 u11 120 0
1039 2 b111 m11 u12 111 3600
1046 2 b111 m11 u12 111 3600
1040
1047
1041 $ hg log -r 'sort(all(), user)'
1048 $ hg log -r 'sort(all(), user)'
1042 3 b112 m111 u11 120 0
1049 3 b112 m111 u11 120 0
1043 1 b11 m12 u111 112 7200
1050 1 b11 m12 u111 112 7200
1044 4 b111 m112 u111 110 14400
1051 4 b111 m112 u111 110 14400
1045 0 b12 m111 u112 111 10800
1052 0 b12 m111 u112 111 10800
1046 2 b111 m11 u12 111 3600
1053 2 b111 m11 u12 111 3600
1047
1054
1048 $ hg log -r 'sort(all(), -user)'
1055 $ hg log -r 'sort(all(), -user)'
1049 2 b111 m11 u12 111 3600
1056 2 b111 m11 u12 111 3600
1050 0 b12 m111 u112 111 10800
1057 0 b12 m111 u112 111 10800
1051 1 b11 m12 u111 112 7200
1058 1 b11 m12 u111 112 7200
1052 4 b111 m112 u111 110 14400
1059 4 b111 m112 u111 110 14400
1053 3 b112 m111 u11 120 0
1060 3 b112 m111 u11 120 0
1054
1061
1055 compare dates (tz offset should have no effect):
1062 compare dates (tz offset should have no effect):
1056
1063
1057 $ hg log -r 'sort(all(), date)'
1064 $ hg log -r 'sort(all(), date)'
1058 4 b111 m112 u111 110 14400
1065 4 b111 m112 u111 110 14400
1059 0 b12 m111 u112 111 10800
1066 0 b12 m111 u112 111 10800
1060 2 b111 m11 u12 111 3600
1067 2 b111 m11 u12 111 3600
1061 1 b11 m12 u111 112 7200
1068 1 b11 m12 u111 112 7200
1062 3 b112 m111 u11 120 0
1069 3 b112 m111 u11 120 0
1063
1070
1064 $ hg log -r 'sort(all(), -date)'
1071 $ hg log -r 'sort(all(), -date)'
1065 3 b112 m111 u11 120 0
1072 3 b112 m111 u11 120 0
1066 1 b11 m12 u111 112 7200
1073 1 b11 m12 u111 112 7200
1067 0 b12 m111 u112 111 10800
1074 0 b12 m111 u112 111 10800
1068 2 b111 m11 u12 111 3600
1075 2 b111 m11 u12 111 3600
1069 4 b111 m112 u111 110 14400
1076 4 b111 m112 u111 110 14400
1070
1077
1071 be aware that 'sort(x, -k)' is not exactly the same as 'reverse(sort(x, k))'
1078 be aware that 'sort(x, -k)' is not exactly the same as 'reverse(sort(x, k))'
1072 because '-k' reverses the comparison, not the list itself:
1079 because '-k' reverses the comparison, not the list itself:
1073
1080
1074 $ hg log -r 'sort(0 + 2, date)'
1081 $ hg log -r 'sort(0 + 2, date)'
1075 0 b12 m111 u112 111 10800
1082 0 b12 m111 u112 111 10800
1076 2 b111 m11 u12 111 3600
1083 2 b111 m11 u12 111 3600
1077
1084
1078 $ hg log -r 'sort(0 + 2, -date)'
1085 $ hg log -r 'sort(0 + 2, -date)'
1079 0 b12 m111 u112 111 10800
1086 0 b12 m111 u112 111 10800
1080 2 b111 m11 u12 111 3600
1087 2 b111 m11 u12 111 3600
1081
1088
1082 $ hg log -r 'reverse(sort(0 + 2, date))'
1089 $ hg log -r 'reverse(sort(0 + 2, date))'
1083 2 b111 m11 u12 111 3600
1090 2 b111 m11 u12 111 3600
1084 0 b12 m111 u112 111 10800
1091 0 b12 m111 u112 111 10800
1085
1092
1086 sort by multiple keys:
1093 sort by multiple keys:
1087
1094
1088 $ hg log -r 'sort(all(), "branch -rev")'
1095 $ hg log -r 'sort(all(), "branch -rev")'
1089 1 b11 m12 u111 112 7200
1096 1 b11 m12 u111 112 7200
1090 4 b111 m112 u111 110 14400
1097 4 b111 m112 u111 110 14400
1091 2 b111 m11 u12 111 3600
1098 2 b111 m11 u12 111 3600
1092 3 b112 m111 u11 120 0
1099 3 b112 m111 u11 120 0
1093 0 b12 m111 u112 111 10800
1100 0 b12 m111 u112 111 10800
1094
1101
1095 $ hg log -r 'sort(all(), "-desc -date")'
1102 $ hg log -r 'sort(all(), "-desc -date")'
1096 1 b11 m12 u111 112 7200
1103 1 b11 m12 u111 112 7200
1097 4 b111 m112 u111 110 14400
1104 4 b111 m112 u111 110 14400
1098 3 b112 m111 u11 120 0
1105 3 b112 m111 u11 120 0
1099 0 b12 m111 u112 111 10800
1106 0 b12 m111 u112 111 10800
1100 2 b111 m11 u12 111 3600
1107 2 b111 m11 u12 111 3600
1101
1108
1102 $ hg log -r 'sort(all(), "user -branch date rev")'
1109 $ hg log -r 'sort(all(), "user -branch date rev")'
1103 3 b112 m111 u11 120 0
1110 3 b112 m111 u11 120 0
1104 4 b111 m112 u111 110 14400
1111 4 b111 m112 u111 110 14400
1105 1 b11 m12 u111 112 7200
1112 1 b11 m12 u111 112 7200
1106 0 b12 m111 u112 111 10800
1113 0 b12 m111 u112 111 10800
1107 2 b111 m11 u12 111 3600
1114 2 b111 m11 u12 111 3600
1108
1115
1109 toposort prioritises graph branches
1116 toposort prioritises graph branches
1110
1117
1111 $ hg up 2
1118 $ hg up 2
1112 0 files updated, 0 files merged, 0 files removed, 0 files unresolved
1119 0 files updated, 0 files merged, 0 files removed, 0 files unresolved
1113 $ touch a
1120 $ touch a
1114 $ hg addremove
1121 $ hg addremove
1115 adding a
1122 adding a
1116 $ hg ci -m 't1' -u 'tu' -d '130 0'
1123 $ hg ci -m 't1' -u 'tu' -d '130 0'
1117 created new head
1124 created new head
1118 $ echo 'a' >> a
1125 $ echo 'a' >> a
1119 $ hg ci -m 't2' -u 'tu' -d '130 0'
1126 $ hg ci -m 't2' -u 'tu' -d '130 0'
1120 $ hg book book1
1127 $ hg book book1
1121 $ hg up 4
1128 $ hg up 4
1122 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
1129 0 files updated, 0 files merged, 1 files removed, 0 files unresolved
1123 (leaving bookmark book1)
1130 (leaving bookmark book1)
1124 $ touch a
1131 $ touch a
1125 $ hg addremove
1132 $ hg addremove
1126 adding a
1133 adding a
1127 $ hg ci -m 't3' -u 'tu' -d '130 0'
1134 $ hg ci -m 't3' -u 'tu' -d '130 0'
1128
1135
1129 $ hg log -r 'sort(all(), topo)'
1136 $ hg log -r 'sort(all(), topo)'
1130 7 b111 t3 tu 130 0
1137 7 b111 t3 tu 130 0
1131 4 b111 m112 u111 110 14400
1138 4 b111 m112 u111 110 14400
1132 3 b112 m111 u11 120 0
1139 3 b112 m111 u11 120 0
1133 6 b111 t2 tu 130 0
1140 6 b111 t2 tu 130 0
1134 5 b111 t1 tu 130 0
1141 5 b111 t1 tu 130 0
1135 2 b111 m11 u12 111 3600
1142 2 b111 m11 u12 111 3600
1136 1 b11 m12 u111 112 7200
1143 1 b11 m12 u111 112 7200
1137 0 b12 m111 u112 111 10800
1144 0 b12 m111 u112 111 10800
1138
1145
1139 $ hg log -r 'sort(all(), -topo)'
1146 $ hg log -r 'sort(all(), -topo)'
1140 0 b12 m111 u112 111 10800
1147 0 b12 m111 u112 111 10800
1141 1 b11 m12 u111 112 7200
1148 1 b11 m12 u111 112 7200
1142 2 b111 m11 u12 111 3600
1149 2 b111 m11 u12 111 3600
1143 5 b111 t1 tu 130 0
1150 5 b111 t1 tu 130 0
1144 6 b111 t2 tu 130 0
1151 6 b111 t2 tu 130 0
1145 3 b112 m111 u11 120 0
1152 3 b112 m111 u11 120 0
1146 4 b111 m112 u111 110 14400
1153 4 b111 m112 u111 110 14400
1147 7 b111 t3 tu 130 0
1154 7 b111 t3 tu 130 0
1148
1155
1149 $ hg log -r 'sort(all(), topo, topo.firstbranch=book1)'
1156 $ hg log -r 'sort(all(), topo, topo.firstbranch=book1)'
1150 6 b111 t2 tu 130 0
1157 6 b111 t2 tu 130 0
1151 5 b111 t1 tu 130 0
1158 5 b111 t1 tu 130 0
1152 7 b111 t3 tu 130 0
1159 7 b111 t3 tu 130 0
1153 4 b111 m112 u111 110 14400
1160 4 b111 m112 u111 110 14400
1154 3 b112 m111 u11 120 0
1161 3 b112 m111 u11 120 0
1155 2 b111 m11 u12 111 3600
1162 2 b111 m11 u12 111 3600
1156 1 b11 m12 u111 112 7200
1163 1 b11 m12 u111 112 7200
1157 0 b12 m111 u112 111 10800
1164 0 b12 m111 u112 111 10800
1158
1165
1159 topographical sorting can't be combined with other sort keys, and you can't
1166 topographical sorting can't be combined with other sort keys, and you can't
1160 use the topo.firstbranch option when topo sort is not active:
1167 use the topo.firstbranch option when topo sort is not active:
1161
1168
1162 $ hg log -r 'sort(all(), "topo user")'
1169 $ hg log -r 'sort(all(), "topo user")'
1163 hg: parse error: topo sort order cannot be combined with other sort keys
1170 hg: parse error: topo sort order cannot be combined with other sort keys
1164 [255]
1171 [255]
1165
1172
1166 $ hg log -r 'sort(all(), user, topo.firstbranch=book1)'
1173 $ hg log -r 'sort(all(), user, topo.firstbranch=book1)'
1167 hg: parse error: topo.firstbranch can only be used when using the topo sort key
1174 hg: parse error: topo.firstbranch can only be used when using the topo sort key
1168 [255]
1175 [255]
1169
1176
1170 $ cd ..
1177 $ cd ..
1171 $ cd repo
1178 $ cd repo
1172
1179
1173 test subtracting something from an addset
1180 test subtracting something from an addset
1174
1181
1175 $ log '(outgoing() or removes(a)) - removes(a)'
1182 $ log '(outgoing() or removes(a)) - removes(a)'
1176 8
1183 8
1177 9
1184 9
1178
1185
1179 test intersecting something with an addset
1186 test intersecting something with an addset
1180
1187
1181 $ log 'parents(outgoing() or removes(a))'
1188 $ log 'parents(outgoing() or removes(a))'
1182 1
1189 1
1183 4
1190 4
1184 5
1191 5
1185 8
1192 8
1186
1193
1187 test that `or` operation combines elements in the right order:
1194 test that `or` operation combines elements in the right order:
1188
1195
1189 $ log '3:4 or 2:5'
1196 $ log '3:4 or 2:5'
1190 3
1197 3
1191 4
1198 4
1192 2
1199 2
1193 5
1200 5
1194 $ log '3:4 or 5:2'
1201 $ log '3:4 or 5:2'
1195 3
1202 3
1196 4
1203 4
1197 5
1204 5
1198 2
1205 2
1199 $ log 'sort(3:4 or 2:5)'
1206 $ log 'sort(3:4 or 2:5)'
1200 2
1207 2
1201 3
1208 3
1202 4
1209 4
1203 5
1210 5
1204 $ log 'sort(3:4 or 5:2)'
1211 $ log 'sort(3:4 or 5:2)'
1205 2
1212 2
1206 3
1213 3
1207 4
1214 4
1208 5
1215 5
1209
1216
1210 test that more than one `-r`s are combined in the right order and deduplicated:
1217 test that more than one `-r`s are combined in the right order and deduplicated:
1211
1218
1212 $ hg log -T '{rev}\n' -r 3 -r 3 -r 4 -r 5:2 -r 'ancestors(4)'
1219 $ hg log -T '{rev}\n' -r 3 -r 3 -r 4 -r 5:2 -r 'ancestors(4)'
1213 3
1220 3
1214 4
1221 4
1215 5
1222 5
1216 2
1223 2
1217 0
1224 0
1218 1
1225 1
1219
1226
1220 test that `or` operation skips duplicated revisions from right-hand side
1227 test that `or` operation skips duplicated revisions from right-hand side
1221
1228
1222 $ try 'reverse(1::5) or ancestors(4)'
1229 $ try 'reverse(1::5) or ancestors(4)'
1223 (or
1230 (or
1224 (func
1231 (func
1225 ('symbol', 'reverse')
1232 ('symbol', 'reverse')
1226 (dagrange
1233 (dagrange
1227 ('symbol', '1')
1234 ('symbol', '1')
1228 ('symbol', '5')))
1235 ('symbol', '5')))
1229 (func
1236 (func
1230 ('symbol', 'ancestors')
1237 ('symbol', 'ancestors')
1231 ('symbol', '4')))
1238 ('symbol', '4')))
1232 * set:
1239 * set:
1233 <addset
1240 <addset
1234 <baseset- [1, 3, 5]>,
1241 <baseset- [1, 3, 5]>,
1235 <generatorset+>>
1242 <generatorset+>>
1236 5
1243 5
1237 3
1244 3
1238 1
1245 1
1239 0
1246 0
1240 2
1247 2
1241 4
1248 4
1242 $ try 'sort(ancestors(4) or reverse(1::5))'
1249 $ try 'sort(ancestors(4) or reverse(1::5))'
1243 (func
1250 (func
1244 ('symbol', 'sort')
1251 ('symbol', 'sort')
1245 (or
1252 (or
1246 (func
1253 (func
1247 ('symbol', 'ancestors')
1254 ('symbol', 'ancestors')
1248 ('symbol', '4'))
1255 ('symbol', '4'))
1249 (func
1256 (func
1250 ('symbol', 'reverse')
1257 ('symbol', 'reverse')
1251 (dagrange
1258 (dagrange
1252 ('symbol', '1')
1259 ('symbol', '1')
1253 ('symbol', '5')))))
1260 ('symbol', '5')))))
1254 * set:
1261 * set:
1255 <addset+
1262 <addset+
1256 <generatorset+>,
1263 <generatorset+>,
1257 <baseset- [1, 3, 5]>>
1264 <baseset- [1, 3, 5]>>
1258 0
1265 0
1259 1
1266 1
1260 2
1267 2
1261 3
1268 3
1262 4
1269 4
1263 5
1270 5
1264
1271
1265 test optimization of trivial `or` operation
1272 test optimization of trivial `or` operation
1266
1273
1267 $ try --optimize '0|(1)|"2"|-2|tip|null'
1274 $ try --optimize '0|(1)|"2"|-2|tip|null'
1268 (or
1275 (or
1269 ('symbol', '0')
1276 ('symbol', '0')
1270 (group
1277 (group
1271 ('symbol', '1'))
1278 ('symbol', '1'))
1272 ('string', '2')
1279 ('string', '2')
1273 (negate
1280 (negate
1274 ('symbol', '2'))
1281 ('symbol', '2'))
1275 ('symbol', 'tip')
1282 ('symbol', 'tip')
1276 ('symbol', 'null'))
1283 ('symbol', 'null'))
1277 * optimized:
1284 * optimized:
1278 (func
1285 (func
1279 ('symbol', '_list')
1286 ('symbol', '_list')
1280 ('string', '0\x001\x002\x00-2\x00tip\x00null'))
1287 ('string', '0\x001\x002\x00-2\x00tip\x00null'))
1281 * set:
1288 * set:
1282 <baseset [0, 1, 2, 8, 9, -1]>
1289 <baseset [0, 1, 2, 8, 9, -1]>
1283 0
1290 0
1284 1
1291 1
1285 2
1292 2
1286 8
1293 8
1287 9
1294 9
1288 -1
1295 -1
1289
1296
1290 $ try --optimize '0|1|2:3'
1297 $ try --optimize '0|1|2:3'
1291 (or
1298 (or
1292 ('symbol', '0')
1299 ('symbol', '0')
1293 ('symbol', '1')
1300 ('symbol', '1')
1294 (range
1301 (range
1295 ('symbol', '2')
1302 ('symbol', '2')
1296 ('symbol', '3')))
1303 ('symbol', '3')))
1297 * optimized:
1304 * optimized:
1298 (or
1305 (or
1299 (func
1306 (func
1300 ('symbol', '_list')
1307 ('symbol', '_list')
1301 ('string', '0\x001'))
1308 ('string', '0\x001'))
1302 (range
1309 (range
1303 ('symbol', '2')
1310 ('symbol', '2')
1304 ('symbol', '3')))
1311 ('symbol', '3')))
1305 * set:
1312 * set:
1306 <addset
1313 <addset
1307 <baseset [0, 1]>,
1314 <baseset [0, 1]>,
1308 <spanset+ 2:3>>
1315 <spanset+ 2:3>>
1309 0
1316 0
1310 1
1317 1
1311 2
1318 2
1312 3
1319 3
1313
1320
1314 $ try --optimize '0:1|2|3:4|5|6'
1321 $ try --optimize '0:1|2|3:4|5|6'
1315 (or
1322 (or
1316 (range
1323 (range
1317 ('symbol', '0')
1324 ('symbol', '0')
1318 ('symbol', '1'))
1325 ('symbol', '1'))
1319 ('symbol', '2')
1326 ('symbol', '2')
1320 (range
1327 (range
1321 ('symbol', '3')
1328 ('symbol', '3')
1322 ('symbol', '4'))
1329 ('symbol', '4'))
1323 ('symbol', '5')
1330 ('symbol', '5')
1324 ('symbol', '6'))
1331 ('symbol', '6'))
1325 * optimized:
1332 * optimized:
1326 (or
1333 (or
1327 (range
1334 (range
1328 ('symbol', '0')
1335 ('symbol', '0')
1329 ('symbol', '1'))
1336 ('symbol', '1'))
1330 ('symbol', '2')
1337 ('symbol', '2')
1331 (range
1338 (range
1332 ('symbol', '3')
1339 ('symbol', '3')
1333 ('symbol', '4'))
1340 ('symbol', '4'))
1334 (func
1341 (func
1335 ('symbol', '_list')
1342 ('symbol', '_list')
1336 ('string', '5\x006')))
1343 ('string', '5\x006')))
1337 * set:
1344 * set:
1338 <addset
1345 <addset
1339 <addset
1346 <addset
1340 <spanset+ 0:1>,
1347 <spanset+ 0:1>,
1341 <baseset [2]>>,
1348 <baseset [2]>>,
1342 <addset
1349 <addset
1343 <spanset+ 3:4>,
1350 <spanset+ 3:4>,
1344 <baseset [5, 6]>>>
1351 <baseset [5, 6]>>>
1345 0
1352 0
1346 1
1353 1
1347 2
1354 2
1348 3
1355 3
1349 4
1356 4
1350 5
1357 5
1351 6
1358 6
1352
1359
1353 test that `_list` should be narrowed by provided `subset`
1360 test that `_list` should be narrowed by provided `subset`
1354
1361
1355 $ log '0:2 and (null|1|2|3)'
1362 $ log '0:2 and (null|1|2|3)'
1356 1
1363 1
1357 2
1364 2
1358
1365
1359 test that `_list` should remove duplicates
1366 test that `_list` should remove duplicates
1360
1367
1361 $ log '0|1|2|1|2|-1|tip'
1368 $ log '0|1|2|1|2|-1|tip'
1362 0
1369 0
1363 1
1370 1
1364 2
1371 2
1365 9
1372 9
1366
1373
1367 test unknown revision in `_list`
1374 test unknown revision in `_list`
1368
1375
1369 $ log '0|unknown'
1376 $ log '0|unknown'
1370 abort: unknown revision 'unknown'!
1377 abort: unknown revision 'unknown'!
1371 [255]
1378 [255]
1372
1379
1373 test integer range in `_list`
1380 test integer range in `_list`
1374
1381
1375 $ log '-1|-10'
1382 $ log '-1|-10'
1376 9
1383 9
1377 0
1384 0
1378
1385
1379 $ log '-10|-11'
1386 $ log '-10|-11'
1380 abort: unknown revision '-11'!
1387 abort: unknown revision '-11'!
1381 [255]
1388 [255]
1382
1389
1383 $ log '9|10'
1390 $ log '9|10'
1384 abort: unknown revision '10'!
1391 abort: unknown revision '10'!
1385 [255]
1392 [255]
1386
1393
1387 test '0000' != '0' in `_list`
1394 test '0000' != '0' in `_list`
1388
1395
1389 $ log '0|0000'
1396 $ log '0|0000'
1390 0
1397 0
1391 -1
1398 -1
1392
1399
1393 test ',' in `_list`
1400 test ',' in `_list`
1394 $ log '0,1'
1401 $ log '0,1'
1395 hg: parse error: can't use a list in this context
1402 hg: parse error: can't use a list in this context
1396 (see hg help "revsets.x or y")
1403 (see hg help "revsets.x or y")
1397 [255]
1404 [255]
1398 $ try '0,1,2'
1405 $ try '0,1,2'
1399 (list
1406 (list
1400 ('symbol', '0')
1407 ('symbol', '0')
1401 ('symbol', '1')
1408 ('symbol', '1')
1402 ('symbol', '2'))
1409 ('symbol', '2'))
1403 hg: parse error: can't use a list in this context
1410 hg: parse error: can't use a list in this context
1404 (see hg help "revsets.x or y")
1411 (see hg help "revsets.x or y")
1405 [255]
1412 [255]
1406
1413
1407 test that chained `or` operations make balanced addsets
1414 test that chained `or` operations make balanced addsets
1408
1415
1409 $ try '0:1|1:2|2:3|3:4|4:5'
1416 $ try '0:1|1:2|2:3|3:4|4:5'
1410 (or
1417 (or
1411 (range
1418 (range
1412 ('symbol', '0')
1419 ('symbol', '0')
1413 ('symbol', '1'))
1420 ('symbol', '1'))
1414 (range
1421 (range
1415 ('symbol', '1')
1422 ('symbol', '1')
1416 ('symbol', '2'))
1423 ('symbol', '2'))
1417 (range
1424 (range
1418 ('symbol', '2')
1425 ('symbol', '2')
1419 ('symbol', '3'))
1426 ('symbol', '3'))
1420 (range
1427 (range
1421 ('symbol', '3')
1428 ('symbol', '3')
1422 ('symbol', '4'))
1429 ('symbol', '4'))
1423 (range
1430 (range
1424 ('symbol', '4')
1431 ('symbol', '4')
1425 ('symbol', '5')))
1432 ('symbol', '5')))
1426 * set:
1433 * set:
1427 <addset
1434 <addset
1428 <addset
1435 <addset
1429 <spanset+ 0:1>,
1436 <spanset+ 0:1>,
1430 <spanset+ 1:2>>,
1437 <spanset+ 1:2>>,
1431 <addset
1438 <addset
1432 <spanset+ 2:3>,
1439 <spanset+ 2:3>,
1433 <addset
1440 <addset
1434 <spanset+ 3:4>,
1441 <spanset+ 3:4>,
1435 <spanset+ 4:5>>>>
1442 <spanset+ 4:5>>>>
1436 0
1443 0
1437 1
1444 1
1438 2
1445 2
1439 3
1446 3
1440 4
1447 4
1441 5
1448 5
1442
1449
1443 no crash by empty group "()" while optimizing `or` operations
1450 no crash by empty group "()" while optimizing `or` operations
1444
1451
1445 $ try --optimize '0|()'
1452 $ try --optimize '0|()'
1446 (or
1453 (or
1447 ('symbol', '0')
1454 ('symbol', '0')
1448 (group
1455 (group
1449 None))
1456 None))
1450 * optimized:
1457 * optimized:
1451 (or
1458 (or
1452 ('symbol', '0')
1459 ('symbol', '0')
1453 None)
1460 None)
1454 hg: parse error: missing argument
1461 hg: parse error: missing argument
1455 [255]
1462 [255]
1456
1463
1457 test that chained `or` operations never eat up stack (issue4624)
1464 test that chained `or` operations never eat up stack (issue4624)
1458 (uses `0:1` instead of `0` to avoid future optimization of trivial revisions)
1465 (uses `0:1` instead of `0` to avoid future optimization of trivial revisions)
1459
1466
1460 $ hg log -T '{rev}\n' -r `python -c "print '+'.join(['0:1'] * 500)"`
1467 $ hg log -T '{rev}\n' -r `python -c "print '+'.join(['0:1'] * 500)"`
1461 0
1468 0
1462 1
1469 1
1463
1470
1464 test that repeated `-r` options never eat up stack (issue4565)
1471 test that repeated `-r` options never eat up stack (issue4565)
1465 (uses `-r 0::1` to avoid possible optimization at old-style parser)
1472 (uses `-r 0::1` to avoid possible optimization at old-style parser)
1466
1473
1467 $ hg log -T '{rev}\n' `python -c "for i in xrange(500): print '-r 0::1 ',"`
1474 $ hg log -T '{rev}\n' `python -c "for i in xrange(500): print '-r 0::1 ',"`
1468 0
1475 0
1469 1
1476 1
1470
1477
1471 check that conversion to only works
1478 check that conversion to only works
1472 $ try --optimize '::3 - ::1'
1479 $ try --optimize '::3 - ::1'
1473 (minus
1480 (minus
1474 (dagrangepre
1481 (dagrangepre
1475 ('symbol', '3'))
1482 ('symbol', '3'))
1476 (dagrangepre
1483 (dagrangepre
1477 ('symbol', '1')))
1484 ('symbol', '1')))
1478 * optimized:
1485 * optimized:
1479 (func
1486 (func
1480 ('symbol', 'only')
1487 ('symbol', 'only')
1481 (list
1488 (list
1482 ('symbol', '3')
1489 ('symbol', '3')
1483 ('symbol', '1')))
1490 ('symbol', '1')))
1484 * set:
1491 * set:
1485 <baseset+ [3]>
1492 <baseset+ [3]>
1486 3
1493 3
1487 $ try --optimize 'ancestors(1) - ancestors(3)'
1494 $ try --optimize 'ancestors(1) - ancestors(3)'
1488 (minus
1495 (minus
1489 (func
1496 (func
1490 ('symbol', 'ancestors')
1497 ('symbol', 'ancestors')
1491 ('symbol', '1'))
1498 ('symbol', '1'))
1492 (func
1499 (func
1493 ('symbol', 'ancestors')
1500 ('symbol', 'ancestors')
1494 ('symbol', '3')))
1501 ('symbol', '3')))
1495 * optimized:
1502 * optimized:
1496 (func
1503 (func
1497 ('symbol', 'only')
1504 ('symbol', 'only')
1498 (list
1505 (list
1499 ('symbol', '1')
1506 ('symbol', '1')
1500 ('symbol', '3')))
1507 ('symbol', '3')))
1501 * set:
1508 * set:
1502 <baseset+ []>
1509 <baseset+ []>
1503 $ try --optimize 'not ::2 and ::6'
1510 $ try --optimize 'not ::2 and ::6'
1504 (and
1511 (and
1505 (not
1512 (not
1506 (dagrangepre
1513 (dagrangepre
1507 ('symbol', '2')))
1514 ('symbol', '2')))
1508 (dagrangepre
1515 (dagrangepre
1509 ('symbol', '6')))
1516 ('symbol', '6')))
1510 * optimized:
1517 * optimized:
1511 (func
1518 (func
1512 ('symbol', 'only')
1519 ('symbol', 'only')
1513 (list
1520 (list
1514 ('symbol', '6')
1521 ('symbol', '6')
1515 ('symbol', '2')))
1522 ('symbol', '2')))
1516 * set:
1523 * set:
1517 <baseset+ [3, 4, 5, 6]>
1524 <baseset+ [3, 4, 5, 6]>
1518 3
1525 3
1519 4
1526 4
1520 5
1527 5
1521 6
1528 6
1522 $ try --optimize 'ancestors(6) and not ancestors(4)'
1529 $ try --optimize 'ancestors(6) and not ancestors(4)'
1523 (and
1530 (and
1524 (func
1531 (func
1525 ('symbol', 'ancestors')
1532 ('symbol', 'ancestors')
1526 ('symbol', '6'))
1533 ('symbol', '6'))
1527 (not
1534 (not
1528 (func
1535 (func
1529 ('symbol', 'ancestors')
1536 ('symbol', 'ancestors')
1530 ('symbol', '4'))))
1537 ('symbol', '4'))))
1531 * optimized:
1538 * optimized:
1532 (func
1539 (func
1533 ('symbol', 'only')
1540 ('symbol', 'only')
1534 (list
1541 (list
1535 ('symbol', '6')
1542 ('symbol', '6')
1536 ('symbol', '4')))
1543 ('symbol', '4')))
1537 * set:
1544 * set:
1538 <baseset+ [3, 5, 6]>
1545 <baseset+ [3, 5, 6]>
1539 3
1546 3
1540 5
1547 5
1541 6
1548 6
1542
1549
1543 no crash by empty group "()" while optimizing to "only()"
1550 no crash by empty group "()" while optimizing to "only()"
1544
1551
1545 $ try --optimize '::1 and ()'
1552 $ try --optimize '::1 and ()'
1546 (and
1553 (and
1547 (dagrangepre
1554 (dagrangepre
1548 ('symbol', '1'))
1555 ('symbol', '1'))
1549 (group
1556 (group
1550 None))
1557 None))
1551 * optimized:
1558 * optimized:
1552 (and
1559 (and
1553 None
1560 None
1554 (func
1561 (func
1555 ('symbol', 'ancestors')
1562 ('symbol', 'ancestors')
1556 ('symbol', '1')))
1563 ('symbol', '1')))
1557 hg: parse error: missing argument
1564 hg: parse error: missing argument
1558 [255]
1565 [255]
1559
1566
1560 we can use patterns when searching for tags
1567 we can use patterns when searching for tags
1561
1568
1562 $ log 'tag("1..*")'
1569 $ log 'tag("1..*")'
1563 abort: tag '1..*' does not exist!
1570 abort: tag '1..*' does not exist!
1564 [255]
1571 [255]
1565 $ log 'tag("re:1..*")'
1572 $ log 'tag("re:1..*")'
1566 6
1573 6
1567 $ log 'tag("re:[0-9].[0-9]")'
1574 $ log 'tag("re:[0-9].[0-9]")'
1568 6
1575 6
1569 $ log 'tag("literal:1.0")'
1576 $ log 'tag("literal:1.0")'
1570 6
1577 6
1571 $ log 'tag("re:0..*")'
1578 $ log 'tag("re:0..*")'
1572
1579
1573 $ log 'tag(unknown)'
1580 $ log 'tag(unknown)'
1574 abort: tag 'unknown' does not exist!
1581 abort: tag 'unknown' does not exist!
1575 [255]
1582 [255]
1576 $ log 'tag("re:unknown")'
1583 $ log 'tag("re:unknown")'
1577 $ log 'present(tag("unknown"))'
1584 $ log 'present(tag("unknown"))'
1578 $ log 'present(tag("re:unknown"))'
1585 $ log 'present(tag("re:unknown"))'
1579 $ log 'branch(unknown)'
1586 $ log 'branch(unknown)'
1580 abort: unknown revision 'unknown'!
1587 abort: unknown revision 'unknown'!
1581 [255]
1588 [255]
1582 $ log 'branch("literal:unknown")'
1589 $ log 'branch("literal:unknown")'
1583 abort: branch 'unknown' does not exist!
1590 abort: branch 'unknown' does not exist!
1584 [255]
1591 [255]
1585 $ log 'branch("re:unknown")'
1592 $ log 'branch("re:unknown")'
1586 $ log 'present(branch("unknown"))'
1593 $ log 'present(branch("unknown"))'
1587 $ log 'present(branch("re:unknown"))'
1594 $ log 'present(branch("re:unknown"))'
1588 $ log 'user(bob)'
1595 $ log 'user(bob)'
1589 2
1596 2
1590
1597
1591 $ log '4::8'
1598 $ log '4::8'
1592 4
1599 4
1593 8
1600 8
1594 $ log '4:8'
1601 $ log '4:8'
1595 4
1602 4
1596 5
1603 5
1597 6
1604 6
1598 7
1605 7
1599 8
1606 8
1600
1607
1601 $ log 'sort(!merge() & (modifies(b) | user(bob) | keyword(bug) | keyword(issue) & 1::9), "-date")'
1608 $ log 'sort(!merge() & (modifies(b) | user(bob) | keyword(bug) | keyword(issue) & 1::9), "-date")'
1602 4
1609 4
1603 2
1610 2
1604 5
1611 5
1605
1612
1606 $ log 'not 0 and 0:2'
1613 $ log 'not 0 and 0:2'
1607 1
1614 1
1608 2
1615 2
1609 $ log 'not 1 and 0:2'
1616 $ log 'not 1 and 0:2'
1610 0
1617 0
1611 2
1618 2
1612 $ log 'not 2 and 0:2'
1619 $ log 'not 2 and 0:2'
1613 0
1620 0
1614 1
1621 1
1615 $ log '(1 and 2)::'
1622 $ log '(1 and 2)::'
1616 $ log '(1 and 2):'
1623 $ log '(1 and 2):'
1617 $ log '(1 and 2):3'
1624 $ log '(1 and 2):3'
1618 $ log 'sort(head(), -rev)'
1625 $ log 'sort(head(), -rev)'
1619 9
1626 9
1620 7
1627 7
1621 6
1628 6
1622 5
1629 5
1623 4
1630 4
1624 3
1631 3
1625 2
1632 2
1626 1
1633 1
1627 0
1634 0
1628 $ log '4::8 - 8'
1635 $ log '4::8 - 8'
1629 4
1636 4
1630
1637
1631 matching() should preserve the order of the input set:
1638 matching() should preserve the order of the input set:
1632
1639
1633 $ log '(2 or 3 or 1) and matching(1 or 2 or 3)'
1640 $ log '(2 or 3 or 1) and matching(1 or 2 or 3)'
1634 2
1641 2
1635 3
1642 3
1636 1
1643 1
1637
1644
1638 $ log 'named("unknown")'
1645 $ log 'named("unknown")'
1639 abort: namespace 'unknown' does not exist!
1646 abort: namespace 'unknown' does not exist!
1640 [255]
1647 [255]
1641 $ log 'named("re:unknown")'
1648 $ log 'named("re:unknown")'
1642 abort: no namespace exists that match 'unknown'!
1649 abort: no namespace exists that match 'unknown'!
1643 [255]
1650 [255]
1644 $ log 'present(named("unknown"))'
1651 $ log 'present(named("unknown"))'
1645 $ log 'present(named("re:unknown"))'
1652 $ log 'present(named("re:unknown"))'
1646
1653
1647 $ log 'tag()'
1654 $ log 'tag()'
1648 6
1655 6
1649 $ log 'named("tags")'
1656 $ log 'named("tags")'
1650 6
1657 6
1651
1658
1652 issue2437
1659 issue2437
1653
1660
1654 $ log '3 and p1(5)'
1661 $ log '3 and p1(5)'
1655 3
1662 3
1656 $ log '4 and p2(6)'
1663 $ log '4 and p2(6)'
1657 4
1664 4
1658 $ log '1 and parents(:2)'
1665 $ log '1 and parents(:2)'
1659 1
1666 1
1660 $ log '2 and children(1:)'
1667 $ log '2 and children(1:)'
1661 2
1668 2
1662 $ log 'roots(all()) or roots(all())'
1669 $ log 'roots(all()) or roots(all())'
1663 0
1670 0
1664 $ hg debugrevspec 'roots(all()) or roots(all())'
1671 $ hg debugrevspec 'roots(all()) or roots(all())'
1665 0
1672 0
1666 $ log 'heads(branch(Γ©)) or heads(branch(Γ©))'
1673 $ log 'heads(branch(Γ©)) or heads(branch(Γ©))'
1667 9
1674 9
1668 $ log 'ancestors(8) and (heads(branch("-a-b-c-")) or heads(branch(Γ©)))'
1675 $ log 'ancestors(8) and (heads(branch("-a-b-c-")) or heads(branch(Γ©)))'
1669 4
1676 4
1670
1677
1671 issue2654: report a parse error if the revset was not completely parsed
1678 issue2654: report a parse error if the revset was not completely parsed
1672
1679
1673 $ log '1 OR 2'
1680 $ log '1 OR 2'
1674 hg: parse error at 2: invalid token
1681 hg: parse error at 2: invalid token
1675 [255]
1682 [255]
1676
1683
1677 or operator should preserve ordering:
1684 or operator should preserve ordering:
1678 $ log 'reverse(2::4) or tip'
1685 $ log 'reverse(2::4) or tip'
1679 4
1686 4
1680 2
1687 2
1681 9
1688 9
1682
1689
1683 parentrevspec
1690 parentrevspec
1684
1691
1685 $ log 'merge()^0'
1692 $ log 'merge()^0'
1686 6
1693 6
1687 $ log 'merge()^'
1694 $ log 'merge()^'
1688 5
1695 5
1689 $ log 'merge()^1'
1696 $ log 'merge()^1'
1690 5
1697 5
1691 $ log 'merge()^2'
1698 $ log 'merge()^2'
1692 4
1699 4
1693 $ log 'merge()^^'
1700 $ log 'merge()^^'
1694 3
1701 3
1695 $ log 'merge()^1^'
1702 $ log 'merge()^1^'
1696 3
1703 3
1697 $ log 'merge()^^^'
1704 $ log 'merge()^^^'
1698 1
1705 1
1699
1706
1700 $ log 'merge()~0'
1707 $ log 'merge()~0'
1701 6
1708 6
1702 $ log 'merge()~1'
1709 $ log 'merge()~1'
1703 5
1710 5
1704 $ log 'merge()~2'
1711 $ log 'merge()~2'
1705 3
1712 3
1706 $ log 'merge()~2^1'
1713 $ log 'merge()~2^1'
1707 1
1714 1
1708 $ log 'merge()~3'
1715 $ log 'merge()~3'
1709 1
1716 1
1710
1717
1711 $ log '(-3:tip)^'
1718 $ log '(-3:tip)^'
1712 4
1719 4
1713 6
1720 6
1714 8
1721 8
1715
1722
1716 $ log 'tip^foo'
1723 $ log 'tip^foo'
1717 hg: parse error: ^ expects a number 0, 1, or 2
1724 hg: parse error: ^ expects a number 0, 1, or 2
1718 [255]
1725 [255]
1719
1726
1720 Bogus function gets suggestions
1727 Bogus function gets suggestions
1721 $ log 'add()'
1728 $ log 'add()'
1722 hg: parse error: unknown identifier: add
1729 hg: parse error: unknown identifier: add
1723 (did you mean adds?)
1730 (did you mean adds?)
1724 [255]
1731 [255]
1725 $ log 'added()'
1732 $ log 'added()'
1726 hg: parse error: unknown identifier: added
1733 hg: parse error: unknown identifier: added
1727 (did you mean adds?)
1734 (did you mean adds?)
1728 [255]
1735 [255]
1729 $ log 'remo()'
1736 $ log 'remo()'
1730 hg: parse error: unknown identifier: remo
1737 hg: parse error: unknown identifier: remo
1731 (did you mean one of remote, removes?)
1738 (did you mean one of remote, removes?)
1732 [255]
1739 [255]
1733 $ log 'babar()'
1740 $ log 'babar()'
1734 hg: parse error: unknown identifier: babar
1741 hg: parse error: unknown identifier: babar
1735 [255]
1742 [255]
1736
1743
1737 Bogus function with a similar internal name doesn't suggest the internal name
1744 Bogus function with a similar internal name doesn't suggest the internal name
1738 $ log 'matches()'
1745 $ log 'matches()'
1739 hg: parse error: unknown identifier: matches
1746 hg: parse error: unknown identifier: matches
1740 (did you mean matching?)
1747 (did you mean matching?)
1741 [255]
1748 [255]
1742
1749
1743 Undocumented functions aren't suggested as similar either
1750 Undocumented functions aren't suggested as similar either
1744 $ log 'wdir2()'
1751 $ log 'wdir2()'
1745 hg: parse error: unknown identifier: wdir2
1752 hg: parse error: unknown identifier: wdir2
1746 [255]
1753 [255]
1747
1754
1748 multiple revspecs
1755 multiple revspecs
1749
1756
1750 $ hg log -r 'tip~1:tip' -r 'tip~2:tip~1' --template '{rev}\n'
1757 $ hg log -r 'tip~1:tip' -r 'tip~2:tip~1' --template '{rev}\n'
1751 8
1758 8
1752 9
1759 9
1753 4
1760 4
1754 5
1761 5
1755 6
1762 6
1756 7
1763 7
1757
1764
1758 test usage in revpair (with "+")
1765 test usage in revpair (with "+")
1759
1766
1760 (real pair)
1767 (real pair)
1761
1768
1762 $ hg diff -r 'tip^^' -r 'tip'
1769 $ hg diff -r 'tip^^' -r 'tip'
1763 diff -r 2326846efdab -r 24286f4ae135 .hgtags
1770 diff -r 2326846efdab -r 24286f4ae135 .hgtags
1764 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1771 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1765 +++ b/.hgtags Thu Jan 01 00:00:00 1970 +0000
1772 +++ b/.hgtags Thu Jan 01 00:00:00 1970 +0000
1766 @@ -0,0 +1,1 @@
1773 @@ -0,0 +1,1 @@
1767 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1774 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1768 $ hg diff -r 'tip^^::tip'
1775 $ hg diff -r 'tip^^::tip'
1769 diff -r 2326846efdab -r 24286f4ae135 .hgtags
1776 diff -r 2326846efdab -r 24286f4ae135 .hgtags
1770 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1777 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1771 +++ b/.hgtags Thu Jan 01 00:00:00 1970 +0000
1778 +++ b/.hgtags Thu Jan 01 00:00:00 1970 +0000
1772 @@ -0,0 +1,1 @@
1779 @@ -0,0 +1,1 @@
1773 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1780 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1774
1781
1775 (single rev)
1782 (single rev)
1776
1783
1777 $ hg diff -r 'tip^' -r 'tip^'
1784 $ hg diff -r 'tip^' -r 'tip^'
1778 $ hg diff -r 'tip^:tip^'
1785 $ hg diff -r 'tip^:tip^'
1779
1786
1780 (single rev that does not looks like a range)
1787 (single rev that does not looks like a range)
1781
1788
1782 $ hg diff -r 'tip^::tip^ or tip^'
1789 $ hg diff -r 'tip^::tip^ or tip^'
1783 diff -r d5d0dcbdc4d9 .hgtags
1790 diff -r d5d0dcbdc4d9 .hgtags
1784 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1791 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1785 +++ b/.hgtags * (glob)
1792 +++ b/.hgtags * (glob)
1786 @@ -0,0 +1,1 @@
1793 @@ -0,0 +1,1 @@
1787 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1794 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1788 $ hg diff -r 'tip^ or tip^'
1795 $ hg diff -r 'tip^ or tip^'
1789 diff -r d5d0dcbdc4d9 .hgtags
1796 diff -r d5d0dcbdc4d9 .hgtags
1790 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1797 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1791 +++ b/.hgtags * (glob)
1798 +++ b/.hgtags * (glob)
1792 @@ -0,0 +1,1 @@
1799 @@ -0,0 +1,1 @@
1793 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1800 +e0cc66ef77e8b6f711815af4e001a6594fde3ba5 1.0
1794
1801
1795 (no rev)
1802 (no rev)
1796
1803
1797 $ hg diff -r 'author("babar") or author("celeste")'
1804 $ hg diff -r 'author("babar") or author("celeste")'
1798 abort: empty revision range
1805 abort: empty revision range
1799 [255]
1806 [255]
1800
1807
1801 aliases:
1808 aliases:
1802
1809
1803 $ echo '[revsetalias]' >> .hg/hgrc
1810 $ echo '[revsetalias]' >> .hg/hgrc
1804 $ echo 'm = merge()' >> .hg/hgrc
1811 $ echo 'm = merge()' >> .hg/hgrc
1805 (revset aliases can override builtin revsets)
1812 (revset aliases can override builtin revsets)
1806 $ echo 'p2($1) = p1($1)' >> .hg/hgrc
1813 $ echo 'p2($1) = p1($1)' >> .hg/hgrc
1807 $ echo 'sincem = descendants(m)' >> .hg/hgrc
1814 $ echo 'sincem = descendants(m)' >> .hg/hgrc
1808 $ echo 'd($1) = reverse(sort($1, date))' >> .hg/hgrc
1815 $ echo 'd($1) = reverse(sort($1, date))' >> .hg/hgrc
1809 $ echo 'rs(ARG1, ARG2) = reverse(sort(ARG1, ARG2))' >> .hg/hgrc
1816 $ echo 'rs(ARG1, ARG2) = reverse(sort(ARG1, ARG2))' >> .hg/hgrc
1810 $ echo 'rs4(ARG1, ARGA, ARGB, ARG2) = reverse(sort(ARG1, ARG2))' >> .hg/hgrc
1817 $ echo 'rs4(ARG1, ARGA, ARGB, ARG2) = reverse(sort(ARG1, ARG2))' >> .hg/hgrc
1811
1818
1812 $ try m
1819 $ try m
1813 ('symbol', 'm')
1820 ('symbol', 'm')
1814 * expanded:
1821 * expanded:
1815 (func
1822 (func
1816 ('symbol', 'merge')
1823 ('symbol', 'merge')
1817 None)
1824 None)
1818 * set:
1825 * set:
1819 <filteredset
1826 <filteredset
1820 <fullreposet+ 0:9>,
1827 <fullreposet+ 0:9>,
1821 <merge>>
1828 <merge>>
1822 6
1829 6
1823
1830
1824 $ HGPLAIN=1
1831 $ HGPLAIN=1
1825 $ export HGPLAIN
1832 $ export HGPLAIN
1826 $ try m
1833 $ try m
1827 ('symbol', 'm')
1834 ('symbol', 'm')
1828 abort: unknown revision 'm'!
1835 abort: unknown revision 'm'!
1829 [255]
1836 [255]
1830
1837
1831 $ HGPLAINEXCEPT=revsetalias
1838 $ HGPLAINEXCEPT=revsetalias
1832 $ export HGPLAINEXCEPT
1839 $ export HGPLAINEXCEPT
1833 $ try m
1840 $ try m
1834 ('symbol', 'm')
1841 ('symbol', 'm')
1835 * expanded:
1842 * expanded:
1836 (func
1843 (func
1837 ('symbol', 'merge')
1844 ('symbol', 'merge')
1838 None)
1845 None)
1839 * set:
1846 * set:
1840 <filteredset
1847 <filteredset
1841 <fullreposet+ 0:9>,
1848 <fullreposet+ 0:9>,
1842 <merge>>
1849 <merge>>
1843 6
1850 6
1844
1851
1845 $ unset HGPLAIN
1852 $ unset HGPLAIN
1846 $ unset HGPLAINEXCEPT
1853 $ unset HGPLAINEXCEPT
1847
1854
1848 $ try 'p2(.)'
1855 $ try 'p2(.)'
1849 (func
1856 (func
1850 ('symbol', 'p2')
1857 ('symbol', 'p2')
1851 ('symbol', '.'))
1858 ('symbol', '.'))
1852 * expanded:
1859 * expanded:
1853 (func
1860 (func
1854 ('symbol', 'p1')
1861 ('symbol', 'p1')
1855 ('symbol', '.'))
1862 ('symbol', '.'))
1856 * set:
1863 * set:
1857 <baseset+ [8]>
1864 <baseset+ [8]>
1858 8
1865 8
1859
1866
1860 $ HGPLAIN=1
1867 $ HGPLAIN=1
1861 $ export HGPLAIN
1868 $ export HGPLAIN
1862 $ try 'p2(.)'
1869 $ try 'p2(.)'
1863 (func
1870 (func
1864 ('symbol', 'p2')
1871 ('symbol', 'p2')
1865 ('symbol', '.'))
1872 ('symbol', '.'))
1866 * set:
1873 * set:
1867 <baseset+ []>
1874 <baseset+ []>
1868
1875
1869 $ HGPLAINEXCEPT=revsetalias
1876 $ HGPLAINEXCEPT=revsetalias
1870 $ export HGPLAINEXCEPT
1877 $ export HGPLAINEXCEPT
1871 $ try 'p2(.)'
1878 $ try 'p2(.)'
1872 (func
1879 (func
1873 ('symbol', 'p2')
1880 ('symbol', 'p2')
1874 ('symbol', '.'))
1881 ('symbol', '.'))
1875 * expanded:
1882 * expanded:
1876 (func
1883 (func
1877 ('symbol', 'p1')
1884 ('symbol', 'p1')
1878 ('symbol', '.'))
1885 ('symbol', '.'))
1879 * set:
1886 * set:
1880 <baseset+ [8]>
1887 <baseset+ [8]>
1881 8
1888 8
1882
1889
1883 $ unset HGPLAIN
1890 $ unset HGPLAIN
1884 $ unset HGPLAINEXCEPT
1891 $ unset HGPLAINEXCEPT
1885
1892
1886 test alias recursion
1893 test alias recursion
1887
1894
1888 $ try sincem
1895 $ try sincem
1889 ('symbol', 'sincem')
1896 ('symbol', 'sincem')
1890 * expanded:
1897 * expanded:
1891 (func
1898 (func
1892 ('symbol', 'descendants')
1899 ('symbol', 'descendants')
1893 (func
1900 (func
1894 ('symbol', 'merge')
1901 ('symbol', 'merge')
1895 None))
1902 None))
1896 * set:
1903 * set:
1897 <addset+
1904 <addset+
1898 <filteredset
1905 <filteredset
1899 <fullreposet+ 0:9>,
1906 <fullreposet+ 0:9>,
1900 <merge>>,
1907 <merge>>,
1901 <generatorset+>>
1908 <generatorset+>>
1902 6
1909 6
1903 7
1910 7
1904
1911
1905 test infinite recursion
1912 test infinite recursion
1906
1913
1907 $ echo 'recurse1 = recurse2' >> .hg/hgrc
1914 $ echo 'recurse1 = recurse2' >> .hg/hgrc
1908 $ echo 'recurse2 = recurse1' >> .hg/hgrc
1915 $ echo 'recurse2 = recurse1' >> .hg/hgrc
1909 $ try recurse1
1916 $ try recurse1
1910 ('symbol', 'recurse1')
1917 ('symbol', 'recurse1')
1911 hg: parse error: infinite expansion of revset alias "recurse1" detected
1918 hg: parse error: infinite expansion of revset alias "recurse1" detected
1912 [255]
1919 [255]
1913
1920
1914 $ echo 'level1($1, $2) = $1 or $2' >> .hg/hgrc
1921 $ echo 'level1($1, $2) = $1 or $2' >> .hg/hgrc
1915 $ echo 'level2($1, $2) = level1($2, $1)' >> .hg/hgrc
1922 $ echo 'level2($1, $2) = level1($2, $1)' >> .hg/hgrc
1916 $ try "level2(level1(1, 2), 3)"
1923 $ try "level2(level1(1, 2), 3)"
1917 (func
1924 (func
1918 ('symbol', 'level2')
1925 ('symbol', 'level2')
1919 (list
1926 (list
1920 (func
1927 (func
1921 ('symbol', 'level1')
1928 ('symbol', 'level1')
1922 (list
1929 (list
1923 ('symbol', '1')
1930 ('symbol', '1')
1924 ('symbol', '2')))
1931 ('symbol', '2')))
1925 ('symbol', '3')))
1932 ('symbol', '3')))
1926 * expanded:
1933 * expanded:
1927 (or
1934 (or
1928 ('symbol', '3')
1935 ('symbol', '3')
1929 (or
1936 (or
1930 ('symbol', '1')
1937 ('symbol', '1')
1931 ('symbol', '2')))
1938 ('symbol', '2')))
1932 * set:
1939 * set:
1933 <addset
1940 <addset
1934 <baseset [3]>,
1941 <baseset [3]>,
1935 <baseset [1, 2]>>
1942 <baseset [1, 2]>>
1936 3
1943 3
1937 1
1944 1
1938 2
1945 2
1939
1946
1940 test nesting and variable passing
1947 test nesting and variable passing
1941
1948
1942 $ echo 'nested($1) = nested2($1)' >> .hg/hgrc
1949 $ echo 'nested($1) = nested2($1)' >> .hg/hgrc
1943 $ echo 'nested2($1) = nested3($1)' >> .hg/hgrc
1950 $ echo 'nested2($1) = nested3($1)' >> .hg/hgrc
1944 $ echo 'nested3($1) = max($1)' >> .hg/hgrc
1951 $ echo 'nested3($1) = max($1)' >> .hg/hgrc
1945 $ try 'nested(2:5)'
1952 $ try 'nested(2:5)'
1946 (func
1953 (func
1947 ('symbol', 'nested')
1954 ('symbol', 'nested')
1948 (range
1955 (range
1949 ('symbol', '2')
1956 ('symbol', '2')
1950 ('symbol', '5')))
1957 ('symbol', '5')))
1951 * expanded:
1958 * expanded:
1952 (func
1959 (func
1953 ('symbol', 'max')
1960 ('symbol', 'max')
1954 (range
1961 (range
1955 ('symbol', '2')
1962 ('symbol', '2')
1956 ('symbol', '5')))
1963 ('symbol', '5')))
1957 * set:
1964 * set:
1958 <baseset
1965 <baseset
1959 <max
1966 <max
1960 <fullreposet+ 0:9>,
1967 <fullreposet+ 0:9>,
1961 <spanset+ 2:5>>>
1968 <spanset+ 2:5>>>
1962 5
1969 5
1963
1970
1964 test chained `or` operations are flattened at parsing phase
1971 test chained `or` operations are flattened at parsing phase
1965
1972
1966 $ echo 'chainedorops($1, $2, $3) = $1|$2|$3' >> .hg/hgrc
1973 $ echo 'chainedorops($1, $2, $3) = $1|$2|$3' >> .hg/hgrc
1967 $ try 'chainedorops(0:1, 1:2, 2:3)'
1974 $ try 'chainedorops(0:1, 1:2, 2:3)'
1968 (func
1975 (func
1969 ('symbol', 'chainedorops')
1976 ('symbol', 'chainedorops')
1970 (list
1977 (list
1971 (range
1978 (range
1972 ('symbol', '0')
1979 ('symbol', '0')
1973 ('symbol', '1'))
1980 ('symbol', '1'))
1974 (range
1981 (range
1975 ('symbol', '1')
1982 ('symbol', '1')
1976 ('symbol', '2'))
1983 ('symbol', '2'))
1977 (range
1984 (range
1978 ('symbol', '2')
1985 ('symbol', '2')
1979 ('symbol', '3'))))
1986 ('symbol', '3'))))
1980 * expanded:
1987 * expanded:
1981 (or
1988 (or
1982 (range
1989 (range
1983 ('symbol', '0')
1990 ('symbol', '0')
1984 ('symbol', '1'))
1991 ('symbol', '1'))
1985 (range
1992 (range
1986 ('symbol', '1')
1993 ('symbol', '1')
1987 ('symbol', '2'))
1994 ('symbol', '2'))
1988 (range
1995 (range
1989 ('symbol', '2')
1996 ('symbol', '2')
1990 ('symbol', '3')))
1997 ('symbol', '3')))
1991 * set:
1998 * set:
1992 <addset
1999 <addset
1993 <spanset+ 0:1>,
2000 <spanset+ 0:1>,
1994 <addset
2001 <addset
1995 <spanset+ 1:2>,
2002 <spanset+ 1:2>,
1996 <spanset+ 2:3>>>
2003 <spanset+ 2:3>>>
1997 0
2004 0
1998 1
2005 1
1999 2
2006 2
2000 3
2007 3
2001
2008
2002 test variable isolation, variable placeholders are rewritten as string
2009 test variable isolation, variable placeholders are rewritten as string
2003 then parsed and matched again as string. Check they do not leak too
2010 then parsed and matched again as string. Check they do not leak too
2004 far away.
2011 far away.
2005
2012
2006 $ echo 'injectparamasstring = max("$1")' >> .hg/hgrc
2013 $ echo 'injectparamasstring = max("$1")' >> .hg/hgrc
2007 $ echo 'callinjection($1) = descendants(injectparamasstring)' >> .hg/hgrc
2014 $ echo 'callinjection($1) = descendants(injectparamasstring)' >> .hg/hgrc
2008 $ try 'callinjection(2:5)'
2015 $ try 'callinjection(2:5)'
2009 (func
2016 (func
2010 ('symbol', 'callinjection')
2017 ('symbol', 'callinjection')
2011 (range
2018 (range
2012 ('symbol', '2')
2019 ('symbol', '2')
2013 ('symbol', '5')))
2020 ('symbol', '5')))
2014 * expanded:
2021 * expanded:
2015 (func
2022 (func
2016 ('symbol', 'descendants')
2023 ('symbol', 'descendants')
2017 (func
2024 (func
2018 ('symbol', 'max')
2025 ('symbol', 'max')
2019 ('string', '$1')))
2026 ('string', '$1')))
2020 abort: unknown revision '$1'!
2027 abort: unknown revision '$1'!
2021 [255]
2028 [255]
2022
2029
2023 test scope of alias expansion: 'universe' is expanded prior to 'shadowall(0)',
2030 test scope of alias expansion: 'universe' is expanded prior to 'shadowall(0)',
2024 but 'all()' should never be substituded to '0()'.
2031 but 'all()' should never be substituded to '0()'.
2025
2032
2026 $ echo 'universe = all()' >> .hg/hgrc
2033 $ echo 'universe = all()' >> .hg/hgrc
2027 $ echo 'shadowall(all) = all and universe' >> .hg/hgrc
2034 $ echo 'shadowall(all) = all and universe' >> .hg/hgrc
2028 $ try 'shadowall(0)'
2035 $ try 'shadowall(0)'
2029 (func
2036 (func
2030 ('symbol', 'shadowall')
2037 ('symbol', 'shadowall')
2031 ('symbol', '0'))
2038 ('symbol', '0'))
2032 * expanded:
2039 * expanded:
2033 (and
2040 (and
2034 ('symbol', '0')
2041 ('symbol', '0')
2035 (func
2042 (func
2036 ('symbol', 'all')
2043 ('symbol', 'all')
2037 None))
2044 None))
2038 * set:
2045 * set:
2039 <filteredset
2046 <filteredset
2040 <baseset [0]>,
2047 <baseset [0]>,
2041 <spanset+ 0:9>>
2048 <spanset+ 0:9>>
2042 0
2049 0
2043
2050
2044 test unknown reference:
2051 test unknown reference:
2045
2052
2046 $ try "unknownref(0)" --config 'revsetalias.unknownref($1)=$1:$2'
2053 $ try "unknownref(0)" --config 'revsetalias.unknownref($1)=$1:$2'
2047 (func
2054 (func
2048 ('symbol', 'unknownref')
2055 ('symbol', 'unknownref')
2049 ('symbol', '0'))
2056 ('symbol', '0'))
2050 abort: bad definition of revset alias "unknownref": invalid symbol '$2'
2057 abort: bad definition of revset alias "unknownref": invalid symbol '$2'
2051 [255]
2058 [255]
2052
2059
2053 $ hg debugrevspec --debug --config revsetalias.anotherbadone='branch(' "tip"
2060 $ hg debugrevspec --debug --config revsetalias.anotherbadone='branch(' "tip"
2054 ('symbol', 'tip')
2061 ('symbol', 'tip')
2055 warning: bad definition of revset alias "anotherbadone": at 7: not a prefix: end
2062 warning: bad definition of revset alias "anotherbadone": at 7: not a prefix: end
2056 * set:
2063 * set:
2057 <baseset [9]>
2064 <baseset [9]>
2058 9
2065 9
2059
2066
2060 $ try 'tip'
2067 $ try 'tip'
2061 ('symbol', 'tip')
2068 ('symbol', 'tip')
2062 * set:
2069 * set:
2063 <baseset [9]>
2070 <baseset [9]>
2064 9
2071 9
2065
2072
2066 $ hg debugrevspec --debug --config revsetalias.'bad name'='tip' "tip"
2073 $ hg debugrevspec --debug --config revsetalias.'bad name'='tip' "tip"
2067 ('symbol', 'tip')
2074 ('symbol', 'tip')
2068 warning: bad declaration of revset alias "bad name": at 4: invalid token
2075 warning: bad declaration of revset alias "bad name": at 4: invalid token
2069 * set:
2076 * set:
2070 <baseset [9]>
2077 <baseset [9]>
2071 9
2078 9
2072 $ echo 'strictreplacing($1, $10) = $10 or desc("$1")' >> .hg/hgrc
2079 $ echo 'strictreplacing($1, $10) = $10 or desc("$1")' >> .hg/hgrc
2073 $ try 'strictreplacing("foo", tip)'
2080 $ try 'strictreplacing("foo", tip)'
2074 (func
2081 (func
2075 ('symbol', 'strictreplacing')
2082 ('symbol', 'strictreplacing')
2076 (list
2083 (list
2077 ('string', 'foo')
2084 ('string', 'foo')
2078 ('symbol', 'tip')))
2085 ('symbol', 'tip')))
2079 * expanded:
2086 * expanded:
2080 (or
2087 (or
2081 ('symbol', 'tip')
2088 ('symbol', 'tip')
2082 (func
2089 (func
2083 ('symbol', 'desc')
2090 ('symbol', 'desc')
2084 ('string', '$1')))
2091 ('string', '$1')))
2085 * set:
2092 * set:
2086 <addset
2093 <addset
2087 <baseset [9]>,
2094 <baseset [9]>,
2088 <filteredset
2095 <filteredset
2089 <fullreposet+ 0:9>,
2096 <fullreposet+ 0:9>,
2090 <desc '$1'>>>
2097 <desc '$1'>>>
2091 9
2098 9
2092
2099
2093 $ try 'd(2:5)'
2100 $ try 'd(2:5)'
2094 (func
2101 (func
2095 ('symbol', 'd')
2102 ('symbol', 'd')
2096 (range
2103 (range
2097 ('symbol', '2')
2104 ('symbol', '2')
2098 ('symbol', '5')))
2105 ('symbol', '5')))
2099 * expanded:
2106 * expanded:
2100 (func
2107 (func
2101 ('symbol', 'reverse')
2108 ('symbol', 'reverse')
2102 (func
2109 (func
2103 ('symbol', 'sort')
2110 ('symbol', 'sort')
2104 (list
2111 (list
2105 (range
2112 (range
2106 ('symbol', '2')
2113 ('symbol', '2')
2107 ('symbol', '5'))
2114 ('symbol', '5'))
2108 ('symbol', 'date'))))
2115 ('symbol', 'date'))))
2109 * set:
2116 * set:
2110 <baseset [4, 5, 3, 2]>
2117 <baseset [4, 5, 3, 2]>
2111 4
2118 4
2112 5
2119 5
2113 3
2120 3
2114 2
2121 2
2115 $ try 'rs(2 or 3, date)'
2122 $ try 'rs(2 or 3, date)'
2116 (func
2123 (func
2117 ('symbol', 'rs')
2124 ('symbol', 'rs')
2118 (list
2125 (list
2119 (or
2126 (or
2120 ('symbol', '2')
2127 ('symbol', '2')
2121 ('symbol', '3'))
2128 ('symbol', '3'))
2122 ('symbol', 'date')))
2129 ('symbol', 'date')))
2123 * expanded:
2130 * expanded:
2124 (func
2131 (func
2125 ('symbol', 'reverse')
2132 ('symbol', 'reverse')
2126 (func
2133 (func
2127 ('symbol', 'sort')
2134 ('symbol', 'sort')
2128 (list
2135 (list
2129 (or
2136 (or
2130 ('symbol', '2')
2137 ('symbol', '2')
2131 ('symbol', '3'))
2138 ('symbol', '3'))
2132 ('symbol', 'date'))))
2139 ('symbol', 'date'))))
2133 * set:
2140 * set:
2134 <baseset [3, 2]>
2141 <baseset [3, 2]>
2135 3
2142 3
2136 2
2143 2
2137 $ try 'rs()'
2144 $ try 'rs()'
2138 (func
2145 (func
2139 ('symbol', 'rs')
2146 ('symbol', 'rs')
2140 None)
2147 None)
2141 hg: parse error: invalid number of arguments: 0
2148 hg: parse error: invalid number of arguments: 0
2142 [255]
2149 [255]
2143 $ try 'rs(2)'
2150 $ try 'rs(2)'
2144 (func
2151 (func
2145 ('symbol', 'rs')
2152 ('symbol', 'rs')
2146 ('symbol', '2'))
2153 ('symbol', '2'))
2147 hg: parse error: invalid number of arguments: 1
2154 hg: parse error: invalid number of arguments: 1
2148 [255]
2155 [255]
2149 $ try 'rs(2, data, 7)'
2156 $ try 'rs(2, data, 7)'
2150 (func
2157 (func
2151 ('symbol', 'rs')
2158 ('symbol', 'rs')
2152 (list
2159 (list
2153 ('symbol', '2')
2160 ('symbol', '2')
2154 ('symbol', 'data')
2161 ('symbol', 'data')
2155 ('symbol', '7')))
2162 ('symbol', '7')))
2156 hg: parse error: invalid number of arguments: 3
2163 hg: parse error: invalid number of arguments: 3
2157 [255]
2164 [255]
2158 $ try 'rs4(2 or 3, x, x, date)'
2165 $ try 'rs4(2 or 3, x, x, date)'
2159 (func
2166 (func
2160 ('symbol', 'rs4')
2167 ('symbol', 'rs4')
2161 (list
2168 (list
2162 (or
2169 (or
2163 ('symbol', '2')
2170 ('symbol', '2')
2164 ('symbol', '3'))
2171 ('symbol', '3'))
2165 ('symbol', 'x')
2172 ('symbol', 'x')
2166 ('symbol', 'x')
2173 ('symbol', 'x')
2167 ('symbol', 'date')))
2174 ('symbol', 'date')))
2168 * expanded:
2175 * expanded:
2169 (func
2176 (func
2170 ('symbol', 'reverse')
2177 ('symbol', 'reverse')
2171 (func
2178 (func
2172 ('symbol', 'sort')
2179 ('symbol', 'sort')
2173 (list
2180 (list
2174 (or
2181 (or
2175 ('symbol', '2')
2182 ('symbol', '2')
2176 ('symbol', '3'))
2183 ('symbol', '3'))
2177 ('symbol', 'date'))))
2184 ('symbol', 'date'))))
2178 * set:
2185 * set:
2179 <baseset [3, 2]>
2186 <baseset [3, 2]>
2180 3
2187 3
2181 2
2188 2
2182
2189
2183 issue4553: check that revset aliases override existing hash prefix
2190 issue4553: check that revset aliases override existing hash prefix
2184
2191
2185 $ hg log -qr e
2192 $ hg log -qr e
2186 6:e0cc66ef77e8
2193 6:e0cc66ef77e8
2187
2194
2188 $ hg log -qr e --config revsetalias.e="all()"
2195 $ hg log -qr e --config revsetalias.e="all()"
2189 0:2785f51eece5
2196 0:2785f51eece5
2190 1:d75937da8da0
2197 1:d75937da8da0
2191 2:5ed5505e9f1c
2198 2:5ed5505e9f1c
2192 3:8528aa5637f2
2199 3:8528aa5637f2
2193 4:2326846efdab
2200 4:2326846efdab
2194 5:904fa392b941
2201 5:904fa392b941
2195 6:e0cc66ef77e8
2202 6:e0cc66ef77e8
2196 7:013af1973af4
2203 7:013af1973af4
2197 8:d5d0dcbdc4d9
2204 8:d5d0dcbdc4d9
2198 9:24286f4ae135
2205 9:24286f4ae135
2199
2206
2200 $ hg log -qr e: --config revsetalias.e="0"
2207 $ hg log -qr e: --config revsetalias.e="0"
2201 0:2785f51eece5
2208 0:2785f51eece5
2202 1:d75937da8da0
2209 1:d75937da8da0
2203 2:5ed5505e9f1c
2210 2:5ed5505e9f1c
2204 3:8528aa5637f2
2211 3:8528aa5637f2
2205 4:2326846efdab
2212 4:2326846efdab
2206 5:904fa392b941
2213 5:904fa392b941
2207 6:e0cc66ef77e8
2214 6:e0cc66ef77e8
2208 7:013af1973af4
2215 7:013af1973af4
2209 8:d5d0dcbdc4d9
2216 8:d5d0dcbdc4d9
2210 9:24286f4ae135
2217 9:24286f4ae135
2211
2218
2212 $ hg log -qr :e --config revsetalias.e="9"
2219 $ hg log -qr :e --config revsetalias.e="9"
2213 0:2785f51eece5
2220 0:2785f51eece5
2214 1:d75937da8da0
2221 1:d75937da8da0
2215 2:5ed5505e9f1c
2222 2:5ed5505e9f1c
2216 3:8528aa5637f2
2223 3:8528aa5637f2
2217 4:2326846efdab
2224 4:2326846efdab
2218 5:904fa392b941
2225 5:904fa392b941
2219 6:e0cc66ef77e8
2226 6:e0cc66ef77e8
2220 7:013af1973af4
2227 7:013af1973af4
2221 8:d5d0dcbdc4d9
2228 8:d5d0dcbdc4d9
2222 9:24286f4ae135
2229 9:24286f4ae135
2223
2230
2224 $ hg log -qr e:
2231 $ hg log -qr e:
2225 6:e0cc66ef77e8
2232 6:e0cc66ef77e8
2226 7:013af1973af4
2233 7:013af1973af4
2227 8:d5d0dcbdc4d9
2234 8:d5d0dcbdc4d9
2228 9:24286f4ae135
2235 9:24286f4ae135
2229
2236
2230 $ hg log -qr :e
2237 $ hg log -qr :e
2231 0:2785f51eece5
2238 0:2785f51eece5
2232 1:d75937da8da0
2239 1:d75937da8da0
2233 2:5ed5505e9f1c
2240 2:5ed5505e9f1c
2234 3:8528aa5637f2
2241 3:8528aa5637f2
2235 4:2326846efdab
2242 4:2326846efdab
2236 5:904fa392b941
2243 5:904fa392b941
2237 6:e0cc66ef77e8
2244 6:e0cc66ef77e8
2238
2245
2239 issue2549 - correct optimizations
2246 issue2549 - correct optimizations
2240
2247
2241 $ try 'limit(1 or 2 or 3, 2) and not 2'
2248 $ try 'limit(1 or 2 or 3, 2) and not 2'
2242 (and
2249 (and
2243 (func
2250 (func
2244 ('symbol', 'limit')
2251 ('symbol', 'limit')
2245 (list
2252 (list
2246 (or
2253 (or
2247 ('symbol', '1')
2254 ('symbol', '1')
2248 ('symbol', '2')
2255 ('symbol', '2')
2249 ('symbol', '3'))
2256 ('symbol', '3'))
2250 ('symbol', '2')))
2257 ('symbol', '2')))
2251 (not
2258 (not
2252 ('symbol', '2')))
2259 ('symbol', '2')))
2253 * set:
2260 * set:
2254 <filteredset
2261 <filteredset
2255 <baseset
2262 <baseset
2256 <limit n=2, offset=0,
2263 <limit n=2, offset=0,
2257 <fullreposet+ 0:9>,
2264 <fullreposet+ 0:9>,
2258 <baseset [1, 2, 3]>>>,
2265 <baseset [1, 2, 3]>>>,
2259 <not
2266 <not
2260 <baseset [2]>>>
2267 <baseset [2]>>>
2261 1
2268 1
2262 $ try 'max(1 or 2) and not 2'
2269 $ try 'max(1 or 2) and not 2'
2263 (and
2270 (and
2264 (func
2271 (func
2265 ('symbol', 'max')
2272 ('symbol', 'max')
2266 (or
2273 (or
2267 ('symbol', '1')
2274 ('symbol', '1')
2268 ('symbol', '2')))
2275 ('symbol', '2')))
2269 (not
2276 (not
2270 ('symbol', '2')))
2277 ('symbol', '2')))
2271 * set:
2278 * set:
2272 <filteredset
2279 <filteredset
2273 <baseset
2280 <baseset
2274 <max
2281 <max
2275 <fullreposet+ 0:9>,
2282 <fullreposet+ 0:9>,
2276 <baseset [1, 2]>>>,
2283 <baseset [1, 2]>>>,
2277 <not
2284 <not
2278 <baseset [2]>>>
2285 <baseset [2]>>>
2279 $ try 'min(1 or 2) and not 1'
2286 $ try 'min(1 or 2) and not 1'
2280 (and
2287 (and
2281 (func
2288 (func
2282 ('symbol', 'min')
2289 ('symbol', 'min')
2283 (or
2290 (or
2284 ('symbol', '1')
2291 ('symbol', '1')
2285 ('symbol', '2')))
2292 ('symbol', '2')))
2286 (not
2293 (not
2287 ('symbol', '1')))
2294 ('symbol', '1')))
2288 * set:
2295 * set:
2289 <filteredset
2296 <filteredset
2290 <baseset
2297 <baseset
2291 <min
2298 <min
2292 <fullreposet+ 0:9>,
2299 <fullreposet+ 0:9>,
2293 <baseset [1, 2]>>>,
2300 <baseset [1, 2]>>>,
2294 <not
2301 <not
2295 <baseset [1]>>>
2302 <baseset [1]>>>
2296 $ try 'last(1 or 2, 1) and not 2'
2303 $ try 'last(1 or 2, 1) and not 2'
2297 (and
2304 (and
2298 (func
2305 (func
2299 ('symbol', 'last')
2306 ('symbol', 'last')
2300 (list
2307 (list
2301 (or
2308 (or
2302 ('symbol', '1')
2309 ('symbol', '1')
2303 ('symbol', '2'))
2310 ('symbol', '2'))
2304 ('symbol', '1')))
2311 ('symbol', '1')))
2305 (not
2312 (not
2306 ('symbol', '2')))
2313 ('symbol', '2')))
2307 * set:
2314 * set:
2308 <filteredset
2315 <filteredset
2309 <baseset
2316 <baseset
2310 <last n=1,
2317 <last n=1,
2311 <fullreposet+ 0:9>,
2318 <fullreposet+ 0:9>,
2312 <baseset [2, 1]>>>,
2319 <baseset [2, 1]>>>,
2313 <not
2320 <not
2314 <baseset [2]>>>
2321 <baseset [2]>>>
2315
2322
2316 issue4289 - ordering of built-ins
2323 issue4289 - ordering of built-ins
2317 $ hg log -M -q -r 3:2
2324 $ hg log -M -q -r 3:2
2318 3:8528aa5637f2
2325 3:8528aa5637f2
2319 2:5ed5505e9f1c
2326 2:5ed5505e9f1c
2320
2327
2321 test revsets started with 40-chars hash (issue3669)
2328 test revsets started with 40-chars hash (issue3669)
2322
2329
2323 $ ISSUE3669_TIP=`hg tip --template '{node}'`
2330 $ ISSUE3669_TIP=`hg tip --template '{node}'`
2324 $ hg log -r "${ISSUE3669_TIP}" --template '{rev}\n'
2331 $ hg log -r "${ISSUE3669_TIP}" --template '{rev}\n'
2325 9
2332 9
2326 $ hg log -r "${ISSUE3669_TIP}^" --template '{rev}\n'
2333 $ hg log -r "${ISSUE3669_TIP}^" --template '{rev}\n'
2327 8
2334 8
2328
2335
2329 test or-ed indirect predicates (issue3775)
2336 test or-ed indirect predicates (issue3775)
2330
2337
2331 $ log '6 or 6^1' | sort
2338 $ log '6 or 6^1' | sort
2332 5
2339 5
2333 6
2340 6
2334 $ log '6^1 or 6' | sort
2341 $ log '6^1 or 6' | sort
2335 5
2342 5
2336 6
2343 6
2337 $ log '4 or 4~1' | sort
2344 $ log '4 or 4~1' | sort
2338 2
2345 2
2339 4
2346 4
2340 $ log '4~1 or 4' | sort
2347 $ log '4~1 or 4' | sort
2341 2
2348 2
2342 4
2349 4
2343 $ log '(0 or 2):(4 or 6) or 0 or 6' | sort
2350 $ log '(0 or 2):(4 or 6) or 0 or 6' | sort
2344 0
2351 0
2345 1
2352 1
2346 2
2353 2
2347 3
2354 3
2348 4
2355 4
2349 5
2356 5
2350 6
2357 6
2351 $ log '0 or 6 or (0 or 2):(4 or 6)' | sort
2358 $ log '0 or 6 or (0 or 2):(4 or 6)' | sort
2352 0
2359 0
2353 1
2360 1
2354 2
2361 2
2355 3
2362 3
2356 4
2363 4
2357 5
2364 5
2358 6
2365 6
2359
2366
2360 tests for 'remote()' predicate:
2367 tests for 'remote()' predicate:
2361 #. (csets in remote) (id) (remote)
2368 #. (csets in remote) (id) (remote)
2362 1. less than local current branch "default"
2369 1. less than local current branch "default"
2363 2. same with local specified "default"
2370 2. same with local specified "default"
2364 3. more than local specified specified
2371 3. more than local specified specified
2365
2372
2366 $ hg clone --quiet -U . ../remote3
2373 $ hg clone --quiet -U . ../remote3
2367 $ cd ../remote3
2374 $ cd ../remote3
2368 $ hg update -q 7
2375 $ hg update -q 7
2369 $ echo r > r
2376 $ echo r > r
2370 $ hg ci -Aqm 10
2377 $ hg ci -Aqm 10
2371 $ log 'remote()'
2378 $ log 'remote()'
2372 7
2379 7
2373 $ log 'remote("a-b-c-")'
2380 $ log 'remote("a-b-c-")'
2374 2
2381 2
2375 $ cd ../repo
2382 $ cd ../repo
2376 $ log 'remote(".a.b.c.", "../remote3")'
2383 $ log 'remote(".a.b.c.", "../remote3")'
2377
2384
2378 tests for concatenation of strings/symbols by "##"
2385 tests for concatenation of strings/symbols by "##"
2379
2386
2380 $ try "278 ## '5f5' ## 1ee ## 'ce5'"
2387 $ try "278 ## '5f5' ## 1ee ## 'ce5'"
2381 (_concat
2388 (_concat
2382 (_concat
2389 (_concat
2383 (_concat
2390 (_concat
2384 ('symbol', '278')
2391 ('symbol', '278')
2385 ('string', '5f5'))
2392 ('string', '5f5'))
2386 ('symbol', '1ee'))
2393 ('symbol', '1ee'))
2387 ('string', 'ce5'))
2394 ('string', 'ce5'))
2388 * concatenated:
2395 * concatenated:
2389 ('string', '2785f51eece5')
2396 ('string', '2785f51eece5')
2390 * set:
2397 * set:
2391 <baseset [0]>
2398 <baseset [0]>
2392 0
2399 0
2393
2400
2394 $ echo 'cat4($1, $2, $3, $4) = $1 ## $2 ## $3 ## $4' >> .hg/hgrc
2401 $ echo 'cat4($1, $2, $3, $4) = $1 ## $2 ## $3 ## $4' >> .hg/hgrc
2395 $ try "cat4(278, '5f5', 1ee, 'ce5')"
2402 $ try "cat4(278, '5f5', 1ee, 'ce5')"
2396 (func
2403 (func
2397 ('symbol', 'cat4')
2404 ('symbol', 'cat4')
2398 (list
2405 (list
2399 ('symbol', '278')
2406 ('symbol', '278')
2400 ('string', '5f5')
2407 ('string', '5f5')
2401 ('symbol', '1ee')
2408 ('symbol', '1ee')
2402 ('string', 'ce5')))
2409 ('string', 'ce5')))
2403 * expanded:
2410 * expanded:
2404 (_concat
2411 (_concat
2405 (_concat
2412 (_concat
2406 (_concat
2413 (_concat
2407 ('symbol', '278')
2414 ('symbol', '278')
2408 ('string', '5f5'))
2415 ('string', '5f5'))
2409 ('symbol', '1ee'))
2416 ('symbol', '1ee'))
2410 ('string', 'ce5'))
2417 ('string', 'ce5'))
2411 * concatenated:
2418 * concatenated:
2412 ('string', '2785f51eece5')
2419 ('string', '2785f51eece5')
2413 * set:
2420 * set:
2414 <baseset [0]>
2421 <baseset [0]>
2415 0
2422 0
2416
2423
2417 (check concatenation in alias nesting)
2424 (check concatenation in alias nesting)
2418
2425
2419 $ echo 'cat2($1, $2) = $1 ## $2' >> .hg/hgrc
2426 $ echo 'cat2($1, $2) = $1 ## $2' >> .hg/hgrc
2420 $ echo 'cat2x2($1, $2, $3, $4) = cat2($1 ## $2, $3 ## $4)' >> .hg/hgrc
2427 $ echo 'cat2x2($1, $2, $3, $4) = cat2($1 ## $2, $3 ## $4)' >> .hg/hgrc
2421 $ log "cat2x2(278, '5f5', 1ee, 'ce5')"
2428 $ log "cat2x2(278, '5f5', 1ee, 'ce5')"
2422 0
2429 0
2423
2430
2424 (check operator priority)
2431 (check operator priority)
2425
2432
2426 $ echo 'cat2n2($1, $2, $3, $4) = $1 ## $2 or $3 ## $4~2' >> .hg/hgrc
2433 $ echo 'cat2n2($1, $2, $3, $4) = $1 ## $2 or $3 ## $4~2' >> .hg/hgrc
2427 $ log "cat2n2(2785f5, 1eece5, 24286f, 4ae135)"
2434 $ log "cat2n2(2785f5, 1eece5, 24286f, 4ae135)"
2428 0
2435 0
2429 4
2436 4
2430
2437
2431 $ cd ..
2438 $ cd ..
2432
2439
2433 prepare repository that has "default" branches of multiple roots
2440 prepare repository that has "default" branches of multiple roots
2434
2441
2435 $ hg init namedbranch
2442 $ hg init namedbranch
2436 $ cd namedbranch
2443 $ cd namedbranch
2437
2444
2438 $ echo default0 >> a
2445 $ echo default0 >> a
2439 $ hg ci -Aqm0
2446 $ hg ci -Aqm0
2440 $ echo default1 >> a
2447 $ echo default1 >> a
2441 $ hg ci -m1
2448 $ hg ci -m1
2442
2449
2443 $ hg branch -q stable
2450 $ hg branch -q stable
2444 $ echo stable2 >> a
2451 $ echo stable2 >> a
2445 $ hg ci -m2
2452 $ hg ci -m2
2446 $ echo stable3 >> a
2453 $ echo stable3 >> a
2447 $ hg ci -m3
2454 $ hg ci -m3
2448
2455
2449 $ hg update -q null
2456 $ hg update -q null
2450 $ echo default4 >> a
2457 $ echo default4 >> a
2451 $ hg ci -Aqm4
2458 $ hg ci -Aqm4
2452 $ echo default5 >> a
2459 $ echo default5 >> a
2453 $ hg ci -m5
2460 $ hg ci -m5
2454
2461
2455 "null" revision belongs to "default" branch (issue4683)
2462 "null" revision belongs to "default" branch (issue4683)
2456
2463
2457 $ log 'branch(null)'
2464 $ log 'branch(null)'
2458 0
2465 0
2459 1
2466 1
2460 4
2467 4
2461 5
2468 5
2462
2469
2463 "null" revision belongs to "default" branch, but it shouldn't appear in set
2470 "null" revision belongs to "default" branch, but it shouldn't appear in set
2464 unless explicitly specified (issue4682)
2471 unless explicitly specified (issue4682)
2465
2472
2466 $ log 'children(branch(default))'
2473 $ log 'children(branch(default))'
2467 1
2474 1
2468 2
2475 2
2469 5
2476 5
2470
2477
2471 $ cd ..
2478 $ cd ..
2472
2479
2473 test author/desc/keyword in problematic encoding
2480 test author/desc/keyword in problematic encoding
2474 # unicode: cp932:
2481 # unicode: cp932:
2475 # u30A2 0x83 0x41(= 'A')
2482 # u30A2 0x83 0x41(= 'A')
2476 # u30C2 0x83 0x61(= 'a')
2483 # u30C2 0x83 0x61(= 'a')
2477
2484
2478 $ hg init problematicencoding
2485 $ hg init problematicencoding
2479 $ cd problematicencoding
2486 $ cd problematicencoding
2480
2487
2481 $ python > setup.sh <<EOF
2488 $ python > setup.sh <<EOF
2482 > print u'''
2489 > print u'''
2483 > echo a > text
2490 > echo a > text
2484 > hg add text
2491 > hg add text
2485 > hg --encoding utf-8 commit -u '\u30A2' -m none
2492 > hg --encoding utf-8 commit -u '\u30A2' -m none
2486 > echo b > text
2493 > echo b > text
2487 > hg --encoding utf-8 commit -u '\u30C2' -m none
2494 > hg --encoding utf-8 commit -u '\u30C2' -m none
2488 > echo c > text
2495 > echo c > text
2489 > hg --encoding utf-8 commit -u none -m '\u30A2'
2496 > hg --encoding utf-8 commit -u none -m '\u30A2'
2490 > echo d > text
2497 > echo d > text
2491 > hg --encoding utf-8 commit -u none -m '\u30C2'
2498 > hg --encoding utf-8 commit -u none -m '\u30C2'
2492 > '''.encode('utf-8')
2499 > '''.encode('utf-8')
2493 > EOF
2500 > EOF
2494 $ sh < setup.sh
2501 $ sh < setup.sh
2495
2502
2496 test in problematic encoding
2503 test in problematic encoding
2497 $ python > test.sh <<EOF
2504 $ python > test.sh <<EOF
2498 > print u'''
2505 > print u'''
2499 > hg --encoding cp932 log --template '{rev}\\n' -r 'author(\u30A2)'
2506 > hg --encoding cp932 log --template '{rev}\\n' -r 'author(\u30A2)'
2500 > echo ====
2507 > echo ====
2501 > hg --encoding cp932 log --template '{rev}\\n' -r 'author(\u30C2)'
2508 > hg --encoding cp932 log --template '{rev}\\n' -r 'author(\u30C2)'
2502 > echo ====
2509 > echo ====
2503 > hg --encoding cp932 log --template '{rev}\\n' -r 'desc(\u30A2)'
2510 > hg --encoding cp932 log --template '{rev}\\n' -r 'desc(\u30A2)'
2504 > echo ====
2511 > echo ====
2505 > hg --encoding cp932 log --template '{rev}\\n' -r 'desc(\u30C2)'
2512 > hg --encoding cp932 log --template '{rev}\\n' -r 'desc(\u30C2)'
2506 > echo ====
2513 > echo ====
2507 > hg --encoding cp932 log --template '{rev}\\n' -r 'keyword(\u30A2)'
2514 > hg --encoding cp932 log --template '{rev}\\n' -r 'keyword(\u30A2)'
2508 > echo ====
2515 > echo ====
2509 > hg --encoding cp932 log --template '{rev}\\n' -r 'keyword(\u30C2)'
2516 > hg --encoding cp932 log --template '{rev}\\n' -r 'keyword(\u30C2)'
2510 > '''.encode('cp932')
2517 > '''.encode('cp932')
2511 > EOF
2518 > EOF
2512 $ sh < test.sh
2519 $ sh < test.sh
2513 0
2520 0
2514 ====
2521 ====
2515 1
2522 1
2516 ====
2523 ====
2517 2
2524 2
2518 ====
2525 ====
2519 3
2526 3
2520 ====
2527 ====
2521 0
2528 0
2522 2
2529 2
2523 ====
2530 ====
2524 1
2531 1
2525 3
2532 3
2526
2533
2527 test error message of bad revset
2534 test error message of bad revset
2528 $ hg log -r 'foo\\'
2535 $ hg log -r 'foo\\'
2529 hg: parse error at 3: syntax error in revset 'foo\\'
2536 hg: parse error at 3: syntax error in revset 'foo\\'
2530 [255]
2537 [255]
2531
2538
2532 $ cd ..
2539 $ cd ..
2533
2540
2534 Test that revset predicate of extension isn't loaded at failure of
2541 Test that revset predicate of extension isn't loaded at failure of
2535 loading it
2542 loading it
2536
2543
2537 $ cd repo
2544 $ cd repo
2538
2545
2539 $ cat <<EOF > $TESTTMP/custompredicate.py
2546 $ cat <<EOF > $TESTTMP/custompredicate.py
2540 > from mercurial import error, registrar, revset
2547 > from mercurial import error, registrar, revset
2541 >
2548 >
2542 > revsetpredicate = registrar.revsetpredicate()
2549 > revsetpredicate = registrar.revsetpredicate()
2543 >
2550 >
2544 > @revsetpredicate('custom1()')
2551 > @revsetpredicate('custom1()')
2545 > def custom1(repo, subset, x):
2552 > def custom1(repo, subset, x):
2546 > return revset.baseset([1])
2553 > return revset.baseset([1])
2547 >
2554 >
2548 > raise error.Abort('intentional failure of loading extension')
2555 > raise error.Abort('intentional failure of loading extension')
2549 > EOF
2556 > EOF
2550 $ cat <<EOF > .hg/hgrc
2557 $ cat <<EOF > .hg/hgrc
2551 > [extensions]
2558 > [extensions]
2552 > custompredicate = $TESTTMP/custompredicate.py
2559 > custompredicate = $TESTTMP/custompredicate.py
2553 > EOF
2560 > EOF
2554
2561
2555 $ hg debugrevspec "custom1()"
2562 $ hg debugrevspec "custom1()"
2556 *** failed to import extension custompredicate from $TESTTMP/custompredicate.py: intentional failure of loading extension
2563 *** failed to import extension custompredicate from $TESTTMP/custompredicate.py: intentional failure of loading extension
2557 hg: parse error: unknown identifier: custom1
2564 hg: parse error: unknown identifier: custom1
2558 [255]
2565 [255]
2559
2566
2560 $ cd ..
2567 $ cd ..
General Comments 0
You need to be logged in to leave comments. Login now