comparison mercurial/revset.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 37b82485097f
children b0790bebfcf8
comparison
equal deleted inserted replaced
34064:8b659b7388c0 34065:c6c8a52e28c9
1575 @predicate('_notpublic', safe=True) 1575 @predicate('_notpublic', safe=True)
1576 def _notpublic(repo, subset, x): 1576 def _notpublic(repo, subset, x):
1577 getargs(x, 0, 0, "_notpublic takes no arguments") 1577 getargs(x, 0, 0, "_notpublic takes no arguments")
1578 return _phase(repo, subset, phases.draft, phases.secret) 1578 return _phase(repo, subset, phases.draft, phases.secret)
1579 1579
1580 # for internal use
1581 @predicate('_phaseandancestors(phasename, set)', safe=True)
1582 def _phaseandancestors(repo, subset, x):
1583 # equivalent to (phasename() & ancestors(set)) but more efficient
1584 # phasename could be one of 'draft', 'secret', or '_notpublic'
1585 args = getargs(x, 2, 2, "_phaseandancestors requires two arguments")
1586 phasename = getsymbol(args[0])
1587 s = getset(repo, fullreposet(repo), args[1])
1588
1589 draft = phases.draft
1590 secret = phases.secret
1591 phasenamemap = {
1592 '_notpublic': draft,
1593 'draft': draft, # follow secret's ancestors
1594 'secret': secret,
1595 }
1596 if phasename not in phasenamemap:
1597 raise error.ParseError('%r is not a valid phasename' % phasename)
1598
1599 minimalphase = phasenamemap[phasename]
1600 getphase = repo._phasecache.phase
1601
1602 def cutfunc(rev):
1603 return getphase(repo, rev) < minimalphase
1604
1605 revs = dagop.revancestors(repo, s, cutfunc=cutfunc)
1606
1607 if phasename == 'draft': # need to remove secret changesets
1608 revs = revs.filter(lambda r: getphase(repo, r) == draft)
1609 return subset & revs
1610
1580 @predicate('public()', safe=True) 1611 @predicate('public()', safe=True)
1581 def public(repo, subset, x): 1612 def public(repo, subset, x):
1582 """Changeset in public phase.""" 1613 """Changeset in public phase."""
1583 # i18n: "public" is a keyword 1614 # i18n: "public" is a keyword
1584 getargs(x, 0, 0, _("public takes no arguments")) 1615 getargs(x, 0, 0, _("public takes no arguments"))