Mercurial > hg
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")) |