comparison mercurial/revsetlang.py @ 34065:c6c8a52e28c9

revset: optimize "draft() & ::x" pattern The `draft() & ::x` type query could be common for selecting one or more draft feature branches being worked on. Before this patch, `::x` may travel through the changelog DAG for a long distance until it gets a smaller revision number than `min(draft())`. It could be very slow on long changelog with distant (in terms of revision numbers) drafts. This patch adds a fast path for this situation, and will stop traveling the changelog DAG once `::x` hits a non-draft revision. The fast path also works for `secret()` and `not public()`. To measure the performance difference, I used drawdag to create a repo that emulates distant drafts: DRAFT4 | DRAFT3 # draft / PUBLIC9999 # public | PUBLIC9998 | . DRAFT2 . | . DRAFT1 # draft | / PUBLIC0001 # public And measured the performance using the repo: (BEFORE) $ hg perfrevset 'draft() & ::(DRAFT2+DRAFT4)' ! wall 0.017132 comb 0.010000 user 0.010000 sys 0.000000 (best of 156) $ hg perfrevset 'draft() & ::(all())' ! wall 0.024221 comb 0.030000 user 0.030000 sys 0.000000 (best of 113) (AFTER) $ hg perfrevset 'draft() & ::(DRAFT2+DRAFT4)' ! wall 0.000243 comb 0.000000 user 0.000000 sys 0.000000 (best of 9303) $ hg perfrevset 'draft() & ::(all())' ! wall 0.004319 comb 0.000000 user 0.000000 sys 0.000000 (best of 655) Differential Revision: https://phab.mercurial-scm.org/D441
author Jun Wu <quark@fb.com>
date Mon, 28 Aug 2017 14:49:00 -0700
parents b2c691d75d93
children e18119b1ad5d
comparison
equal deleted inserted replaced
34064:8b659b7388c0 34065:c6c8a52e28c9
366 return smallbonus, x # single revisions are small 366 return smallbonus, x # single revisions are small
367 elif op == 'and': 367 elif op == 'and':
368 wa, ta = _optimize(x[1], True) 368 wa, ta = _optimize(x[1], True)
369 wb, tb = _optimize(x[2], True) 369 wb, tb = _optimize(x[2], True)
370 w = min(wa, wb) 370 w = min(wa, wb)
371
372 # (draft/secret/_notpublic() & ::x) have a fast path
373 m = _match('_() & ancestors(_)', ('and', ta, tb))
374 if m and getsymbol(m[1]) in {'draft', 'secret', '_notpublic'}:
375 return w, _build('_phaseandancestors(_, _)', m[1], m[2])
371 376
372 # (::x and not ::y)/(not ::y and ::x) have a fast path 377 # (::x and not ::y)/(not ::y and ::x) have a fast path
373 m = _matchonly(ta, tb) or _matchonly(tb, ta) 378 m = _matchonly(ta, tb) or _matchonly(tb, ta)
374 if m: 379 if m:
375 return w, _build('only(_, _)', *m[1:]) 380 return w, _build('only(_, _)', *m[1:])