bookmarks.py
573 lines
| 20.3 KiB
| text/x-python
|
PythonLexer
/ hgext / bookmarks.py
David Soria Parra
|
r7239 | # Mercurial extension to provide the 'hg bookmark' command | ||
# | ||||
# Copyright 2008 David Soria Parra <dsp@php.net> | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Joel Rosdahl
|
r7252 | |||
Cédric Duval
|
r8894 | '''track a line of development with movable markers | ||
David Soria Parra
|
r7239 | |||
Martin Geisler
|
r9251 | Bookmarks are local movable markers to changesets. Every bookmark | ||
points to a changeset identified by its hash. If you commit a | ||||
changeset that is based on a changeset that has a bookmark on it, the | ||||
bookmark shifts to the new changeset. | ||||
David Soria Parra
|
r7239 | |||
Martin Geisler
|
r11193 | It is possible to use bookmark names in every revision lookup (e.g. | ||
:hg:`merge`, :hg:`update`). | ||||
David Soria Parra
|
r7481 | |||
Martin Geisler
|
r9251 | By default, when several bookmarks point to the same changeset, they | ||
will all move forward together. It is possible to obtain a more | ||||
git-like experience by adding the following configuration option to | ||||
Brodie Rao
|
r12083 | your configuration file:: | ||
David Soria Parra
|
r7481 | |||
Cédric Duval
|
r8892 | [bookmarks] | ||
track.current = True | ||||
David Soria Parra
|
r7481 | |||
Martin Geisler
|
r9251 | This will cause Mercurial to track the bookmark that you are currently | ||
using, and only update it. This is similar to git's approach to | ||||
branching. | ||||
David Soria Parra
|
r7239 | ''' | ||
Joel Rosdahl
|
r7252 | |||
David Soria Parra
|
r7239 | from mercurial.i18n import _ | ||
Augie Fackler
|
r12714 | from mercurial.node import nullid, nullrev, bin, hex, short | ||
David Soria Parra
|
r11431 | from mercurial import util, commands, repair, extensions, pushkey, hg, url | ||
Augie Fackler
|
r12714 | from mercurial import revset | ||
Matt Mackall
|
r7638 | import os | ||
David Soria Parra
|
r7239 | |||
Nicolas Dumazet
|
r10106 | def write(repo): | ||
David Soria Parra
|
r7239 | '''Write bookmarks | ||
Joel Rosdahl
|
r7250 | |||
David Soria Parra
|
r7239 | Write the given bookmark => hash dictionary to the .hg/bookmarks file | ||
in a format equal to those of localtags. | ||||
We also store a backup of the previous state in undo.bookmarks that | ||||
can be copied back on rollback. | ||||
''' | ||||
Nicolas Dumazet
|
r10106 | refs = repo._bookmarks | ||
Joel Rosdahl
|
r7253 | if os.path.exists(repo.join('bookmarks')): | ||
util.copyfile(repo.join('bookmarks'), repo.join('undo.bookmarks')) | ||||
Nicolas Dumazet
|
r10107 | if repo._bookmarkcurrent not in refs: | ||
David Soria Parra
|
r7481 | setcurrent(repo, None) | ||
Matt Mackall
|
r8862 | wlock = repo.wlock() | ||
try: | ||||
file = repo.opener('bookmarks', 'w', atomictemp=True) | ||||
for refspec, node in refs.iteritems(): | ||||
file.write("%s %s\n" % (hex(node), refspec)) | ||||
file.rename() | ||||
Matt Mackall
|
r11440 | |||
# touch 00changelog.i so hgweb reloads bookmarks (no lock needed) | ||||
try: | ||||
os.utime(repo.sjoin('00changelog.i'), None) | ||||
except OSError: | ||||
pass | ||||
Matt Mackall
|
r8862 | finally: | ||
wlock.release() | ||||
David Soria Parra
|
r7239 | |||
David Soria Parra
|
r7481 | def setcurrent(repo, mark): | ||
'''Set the name of the bookmark that we are currently on | ||||
Set the name of the bookmark that we are on (hg update <bookmark>). | ||||
Wagner Bruna
|
r8087 | The name is recorded in .hg/bookmarks.current | ||
David Soria Parra
|
r7481 | ''' | ||
Nicolas Dumazet
|
r10107 | current = repo._bookmarkcurrent | ||
if current == mark: | ||||
David Soria Parra
|
r7481 | return | ||
David Soria Parra
|
r7484 | |||
Nicolas Dumazet
|
r10105 | refs = repo._bookmarks | ||
David Soria Parra
|
r7484 | |||
Benoit Boissinot
|
r7491 | # do not update if we do update to a rev equal to the current bookmark | ||
Alex Unden
|
r7817 | if (mark and mark not in refs and | ||
Nicolas Dumazet
|
r10107 | current and refs[current] == repo.changectx('.').node()): | ||
David Soria Parra
|
r7484 | return | ||
David Soria Parra
|
r7481 | if mark not in refs: | ||
mark = '' | ||||
Matt Mackall
|
r8862 | wlock = repo.wlock() | ||
try: | ||||
file = repo.opener('bookmarks.current', 'w', atomictemp=True) | ||||
file.write(mark) | ||||
file.rename() | ||||
finally: | ||||
wlock.release() | ||||
David Soria Parra
|
r7481 | repo._bookmarkcurrent = mark | ||
Joel Rosdahl
|
r7255 | def bookmark(ui, repo, mark=None, rev=None, force=False, delete=False, rename=None): | ||
Cédric Duval
|
r8894 | '''track a line of development with movable markers | ||
Joel Rosdahl
|
r7250 | |||
Martin Geisler
|
r9251 | Bookmarks are pointers to certain commits that move when | ||
committing. Bookmarks are local. They can be renamed, copied and | ||||
Martin Geisler
|
r10973 | deleted. It is possible to use bookmark names in :hg:`merge` and | ||
:hg:`update` to merge and update respectively to a given bookmark. | ||||
Joel Rosdahl
|
r7250 | |||
Martin Geisler
|
r10973 | You can use :hg:`bookmark NAME` to set a bookmark on the working | ||
Martin Geisler
|
r9251 | directory's parent revision with the given name. If you specify | ||
a revision using -r REV (where REV may be an existing bookmark), | ||||
the bookmark is assigned to that revision. | ||||
Kevin Bullock
|
r12772 | |||
Bookmarks can be pushed and pulled between repositories (see :hg:`help | ||||
push` and :hg:`help pull`). This requires the bookmark extension to be | ||||
enabled for both the local and remote repositories. | ||||
David Soria Parra
|
r7239 | ''' | ||
hexfn = ui.debugflag and hex or short | ||||
Nicolas Dumazet
|
r10105 | marks = repo._bookmarks | ||
David Soria Parra
|
r7239 | cur = repo.changectx('.').node() | ||
Joel Rosdahl
|
r7255 | if rename: | ||
if rename not in marks: | ||||
Joel Rosdahl
|
r7251 | raise util.Abort(_("a bookmark of this name does not exist")) | ||
David Soria Parra
|
r7239 | if mark in marks and not force: | ||
raise util.Abort(_("a bookmark of the same name already exists")) | ||||
Joel Rosdahl
|
r7254 | if mark is None: | ||
raise util.Abort(_("new bookmark name required")) | ||||
Joel Rosdahl
|
r7255 | marks[mark] = marks[rename] | ||
del marks[rename] | ||||
Nicolas Dumazet
|
r10107 | if repo._bookmarkcurrent == rename: | ||
David Soria Parra
|
r7550 | setcurrent(repo, mark) | ||
Nicolas Dumazet
|
r10106 | write(repo) | ||
David Soria Parra
|
r7239 | return | ||
Joel Rosdahl
|
r7250 | |||
David Soria Parra
|
r7239 | if delete: | ||
Martin Geisler
|
r8527 | if mark is None: | ||
David Soria Parra
|
r7239 | raise util.Abort(_("bookmark name required")) | ||
if mark not in marks: | ||||
Joel Rosdahl
|
r7251 | raise util.Abort(_("a bookmark of this name does not exist")) | ||
Nicolas Dumazet
|
r10107 | if mark == repo._bookmarkcurrent: | ||
Alex Unden
|
r7817 | setcurrent(repo, None) | ||
David Soria Parra
|
r7239 | del marks[mark] | ||
Nicolas Dumazet
|
r10106 | write(repo) | ||
David Soria Parra
|
r7239 | return | ||
if mark != None: | ||||
Joel Rosdahl
|
r7259 | if "\n" in mark: | ||
raise util.Abort(_("bookmark name cannot contain newlines")) | ||||
Joel Rosdahl
|
r7260 | mark = mark.strip() | ||
Idan Kamara
|
r11704 | if not mark: | ||
raise util.Abort(_("bookmark names cannot consist entirely of " | ||||
"whitespace")) | ||||
David Soria Parra
|
r7239 | if mark in marks and not force: | ||
raise util.Abort(_("a bookmark of the same name already exists")) | ||||
Joel Rosdahl
|
r7250 | if ((mark in repo.branchtags() or mark == repo.dirstate.branch()) | ||
David Soria Parra
|
r7239 | and not force): | ||
Joel Rosdahl
|
r7252 | raise util.Abort( | ||
_("a bookmark cannot have the name of an existing branch")) | ||||
David Soria Parra
|
r7239 | if rev: | ||
marks[mark] = repo.lookup(rev) | ||||
else: | ||||
marks[mark] = repo.changectx('.').node() | ||||
Brodie Rao
|
r11561 | setcurrent(repo, mark) | ||
Nicolas Dumazet
|
r10106 | write(repo) | ||
David Soria Parra
|
r7239 | return | ||
Martin Geisler
|
r8527 | if mark is None: | ||
Joel Rosdahl
|
r7258 | if rev: | ||
raise util.Abort(_("bookmark name required")) | ||||
David Soria Parra
|
r7239 | if len(marks) == 0: | ||
Benoit Boissinot
|
r10510 | ui.status(_("no bookmarks set\n")) | ||
David Soria Parra
|
r7239 | else: | ||
for bmark, n in marks.iteritems(): | ||||
David Soria Parra
|
r7481 | if ui.configbool('bookmarks', 'track.current'): | ||
Nicolas Dumazet
|
r10107 | current = repo._bookmarkcurrent | ||
Brodie Rao
|
r10820 | if bmark == current and n == cur: | ||
prefix, label = '*', 'bookmarks.current' | ||||
else: | ||||
prefix, label = ' ', '' | ||||
David Soria Parra
|
r7481 | else: | ||
Brodie Rao
|
r10820 | if n == cur: | ||
prefix, label = '*', 'bookmarks.current' | ||||
else: | ||||
prefix, label = ' ', '' | ||||
David Soria Parra
|
r7481 | |||
Steve Losh
|
r9459 | if ui.quiet: | ||
Brodie Rao
|
r10820 | ui.write("%s\n" % bmark, label=label) | ||
Steve Losh
|
r9459 | else: | ||
ui.write(" %s %-25s %d:%s\n" % ( | ||||
Brodie Rao
|
r10820 | prefix, bmark, repo.changelog.rev(n), hexfn(n)), | ||
label=label) | ||||
David Soria Parra
|
r7239 | return | ||
def _revstostrip(changelog, node): | ||||
srev = changelog.rev(node) | ||||
tostrip = [srev] | ||||
saveheads = [] | ||||
Benoit Boissinot
|
r7283 | for r in xrange(srev, len(changelog)): | ||
David Soria Parra
|
r7239 | parents = changelog.parentrevs(r) | ||
if parents[0] in tostrip or parents[1] in tostrip: | ||||
tostrip.append(r) | ||||
if parents[1] != nullrev: | ||||
for p in parents: | ||||
Benoit Boissinot
|
r7283 | if p not in tostrip and p > srev: | ||
David Soria Parra
|
r7239 | saveheads.append(p) | ||
return [r for r in tostrip if r not in saveheads] | ||||
Matt Mackall
|
r7638 | def strip(oldstrip, ui, repo, node, backup="all"): | ||
David Soria Parra
|
r7239 | """Strip bookmarks if revisions are stripped using | ||
the mercurial.strip method. This usually happens during | ||||
qpush and qpop""" | ||||
revisions = _revstostrip(repo.changelog, node) | ||||
Nicolas Dumazet
|
r10105 | marks = repo._bookmarks | ||
David Soria Parra
|
r7239 | update = [] | ||
Dirkjan Ochtman
|
r7622 | for mark, n in marks.iteritems(): | ||
David Soria Parra
|
r7239 | if repo.changelog.rev(n) in revisions: | ||
update.append(mark) | ||||
Benoit Boissinot
|
r7280 | oldstrip(ui, repo, node, backup) | ||
David Soria Parra
|
r7239 | if len(update) > 0: | ||
for m in update: | ||||
marks[m] = repo.changectx('.').node() | ||||
Nicolas Dumazet
|
r10106 | write(repo) | ||
David Soria Parra
|
r7239 | |||
def reposetup(ui, repo): | ||||
Alexander Solovyov
|
r9643 | if not repo.local(): | ||
David Soria Parra
|
r7239 | return | ||
class bookmark_repo(repo.__class__): | ||||
Nicolas Dumazet
|
r10105 | |||
@util.propertycache | ||||
def _bookmarks(self): | ||||
Nicolas Dumazet
|
r10109 | '''Parse .hg/bookmarks file and return a dictionary | ||
Bookmarks are stored as {HASH}\\s{NAME}\\n (localtags format) values | ||||
Nicolas Dumazet
|
r11641 | in the .hg/bookmarks file. | ||
Read the file and return a (name=>nodeid) dictionary | ||||
Nicolas Dumazet
|
r10109 | ''' | ||
try: | ||||
bookmarks = {} | ||||
for line in self.opener('bookmarks'): | ||||
sha, refspec = line.strip().split(' ', 1) | ||||
Paul Molodowitch
|
r12747 | bookmarks[refspec] = self.changelog.lookup(sha) | ||
Nicolas Dumazet
|
r10109 | except: | ||
pass | ||||
return bookmarks | ||||
David Soria Parra
|
r7239 | |||
Nicolas Dumazet
|
r10107 | @util.propertycache | ||
def _bookmarkcurrent(self): | ||||
Nicolas Dumazet
|
r10109 | '''Get the current bookmark | ||
If we use gittishsh branches we have a current bookmark that | ||||
we are on. This function returns the name of the bookmark. It | ||||
is stored in .hg/bookmarks.current | ||||
''' | ||||
mark = None | ||||
if os.path.exists(self.join('bookmarks.current')): | ||||
file = self.opener('bookmarks.current') | ||||
# No readline() in posixfile_nt, reading everything is cheap | ||||
mark = (file.readlines() or [''])[0] | ||||
if mark == '': | ||||
mark = None | ||||
file.close() | ||||
return mark | ||||
Nicolas Dumazet
|
r10107 | |||
Steve Borho
|
r10882 | def rollback(self, *args): | ||
David Soria Parra
|
r7239 | if os.path.exists(self.join('undo.bookmarks')): | ||
util.rename(self.join('undo.bookmarks'), self.join('bookmarks')) | ||||
Steve Borho
|
r10882 | return super(bookmark_repo, self).rollback(*args) | ||
David Soria Parra
|
r7239 | |||
def lookup(self, key): | ||||
Paul Molodowitch
|
r12747 | if key in self._bookmarks: | ||
key = self._bookmarks[key] | ||||
David Soria Parra
|
r7239 | return super(bookmark_repo, self).lookup(key) | ||
Nicolas Dumazet
|
r10108 | def _bookmarksupdate(self, parents, node): | ||
Nicolas Dumazet
|
r10105 | marks = self._bookmarks | ||
David Soria Parra
|
r7239 | update = False | ||
David Soria Parra
|
r9236 | if ui.configbool('bookmarks', 'track.current'): | ||
Nicolas Dumazet
|
r10107 | mark = self._bookmarkcurrent | ||
David Soria Parra
|
r9236 | if mark and marks[mark] in parents: | ||
David Soria Parra
|
r7239 | marks[mark] = node | ||
update = True | ||||
David Soria Parra
|
r9236 | else: | ||
for mark, n in marks.items(): | ||||
if n in parents: | ||||
marks[mark] = node | ||||
update = True | ||||
David Soria Parra
|
r7239 | if update: | ||
Nicolas Dumazet
|
r10106 | write(self) | ||
Nicolas Dumazet
|
r10108 | |||
David Soria Parra
|
r9235 | def commitctx(self, ctx, error=False): | ||
David Soria Parra
|
r7239 | """Add a revision to the repository and | ||
move the bookmark""" | ||||
Matt Mackall
|
r8862 | wlock = self.wlock() # do both commit and bookmark with lock held | ||
try: | ||||
David Soria Parra
|
r9235 | node = super(bookmark_repo, self).commitctx(ctx, error) | ||
Matt Mackall
|
r8862 | if node is None: | ||
return None | ||||
Isaac Jurado
|
r8944 | parents = self.changelog.parents(node) | ||
Matt Mackall
|
r8862 | if parents[1] == nullid: | ||
parents = (parents[0],) | ||||
Nicolas Dumazet
|
r10108 | |||
self._bookmarksupdate(parents, node) | ||||
Matt Mackall
|
r8862 | return node | ||
finally: | ||||
wlock.release() | ||||
David Soria Parra
|
r7239 | |||
Matt Mackall
|
r11373 | def pull(self, remote, heads=None, force=False): | ||
result = super(bookmark_repo, self).pull(remote, heads, force) | ||||
self.ui.debug("checking for updated bookmarks\n") | ||||
rb = remote.listkeys('bookmarks') | ||||
Martin Geisler
|
r11774 | changed = False | ||
Matt Mackall
|
r11373 | for k in rb.keys(): | ||
if k in self._bookmarks: | ||||
nr, nl = rb[k], self._bookmarks[k] | ||||
if nr in self: | ||||
cr = self[nr] | ||||
cl = self[nl] | ||||
if cl.rev() >= cr.rev(): | ||||
continue | ||||
if cr in cl.descendants(): | ||||
self._bookmarks[k] = cr.node() | ||||
Martin Geisler
|
r11774 | changed = True | ||
Matt Mackall
|
r11373 | self.ui.status(_("updating bookmark %s\n") % k) | ||
else: | ||||
self.ui.warn(_("not updating divergent" | ||||
" bookmark %s\n") % k) | ||||
Martin Geisler
|
r11774 | if changed: | ||
Matt Mackall
|
r11373 | write(repo) | ||
return result | ||||
Matt Mackall
|
r11374 | def push(self, remote, force=False, revs=None, newbranch=False): | ||
result = super(bookmark_repo, self).push(remote, force, revs, | ||||
newbranch) | ||||
self.ui.debug("checking for updated bookmarks\n") | ||||
rb = remote.listkeys('bookmarks') | ||||
for k in rb.keys(): | ||||
if k in self._bookmarks: | ||||
nr, nl = rb[k], self._bookmarks[k] | ||||
if nr in self: | ||||
cr = self[nr] | ||||
cl = self[nl] | ||||
if cl in cr.descendants(): | ||||
r = remote.pushkey('bookmarks', k, nr, nl) | ||||
if r: | ||||
self.ui.status(_("updating bookmark %s\n") % k) | ||||
else: | ||||
David Soria Parra
|
r11434 | self.ui.warn(_('updating bookmark %s' | ||
' failed!\n') % k) | ||||
Matt Mackall
|
r11374 | |||
return result | ||||
Matt Mackall
|
r11442 | def addchangegroup(self, *args, **kwargs): | ||
Isaac Jurado
|
r8944 | parents = self.dirstate.parents() | ||
David Soria Parra
|
r7239 | |||
Matt Mackall
|
r11442 | result = super(bookmark_repo, self).addchangegroup(*args, **kwargs) | ||
David Soria Parra
|
r7239 | if result > 1: | ||
# We have more heads than before | ||||
return result | ||||
Isaac Jurado
|
r8944 | node = self.changelog.tip() | ||
Nicolas Dumazet
|
r10108 | |||
self._bookmarksupdate(parents, node) | ||||
David Soria Parra
|
r7239 | return result | ||
Greg Ward
|
r9145 | def _findtags(self): | ||
Martin Geisler
|
r7480 | """Merge bookmarks with normal tags""" | ||
Greg Ward
|
r9145 | (tags, tagtypes) = super(bookmark_repo, self)._findtags() | ||
Paul Molodowitch
|
r12747 | tags.update(self._bookmarks) | ||
Greg Ward
|
r9145 | return (tags, tagtypes) | ||
Martin Geisler
|
r7480 | |||
Paul Molodowitch
|
r10597 | if hasattr(repo, 'invalidate'): | ||
def invalidate(self): | ||||
super(bookmark_repo, self).invalidate() | ||||
for attr in ('_bookmarks', '_bookmarkcurrent'): | ||||
if attr in self.__dict__: | ||||
Isaac Jurado
|
r10956 | delattr(self, attr) | ||
Paul Molodowitch
|
r10597 | |||
David Soria Parra
|
r7239 | repo.__class__ = bookmark_repo | ||
Matt Mackall
|
r11372 | def listbookmarks(repo): | ||
Martin Geisler
|
r12026 | # We may try to list bookmarks on a repo type that does not | ||
# support it (e.g., statichttprepository). | ||||
if not hasattr(repo, '_bookmarks'): | ||||
return {} | ||||
Matt Mackall
|
r11372 | d = {} | ||
for k, v in repo._bookmarks.iteritems(): | ||||
d[k] = hex(v) | ||||
return d | ||||
def pushbookmark(repo, key, old, new): | ||||
w = repo.wlock() | ||||
try: | ||||
marks = repo._bookmarks | ||||
if hex(marks.get(key, '')) != old: | ||||
return False | ||||
if new == '': | ||||
del marks[key] | ||||
else: | ||||
if new not in repo: | ||||
return False | ||||
marks[key] = repo[new].node() | ||||
write(repo) | ||||
return True | ||||
finally: | ||||
w.release() | ||||
Matt Mackall
|
r11378 | def pull(oldpull, ui, repo, source="default", **opts): | ||
# translate bookmark args to rev args for actual pull | ||||
if opts.get('bookmark'): | ||||
# this is an unpleasant hack as pull will do this internally | ||||
source, branches = hg.parseurl(ui.expandpath(source), | ||||
opts.get('branch')) | ||||
other = hg.repository(hg.remoteui(repo, opts), source) | ||||
rb = other.listkeys('bookmarks') | ||||
for b in opts['bookmark']: | ||||
if b not in rb: | ||||
raise util.Abort(_('remote bookmark %s not found!') % b) | ||||
opts.setdefault('rev', []).append(b) | ||||
result = oldpull(ui, repo, source, **opts) | ||||
# update specified bookmarks | ||||
if opts.get('bookmark'): | ||||
for b in opts['bookmark']: | ||||
# explicit pull overrides local bookmark if any | ||||
ui.status(_("importing bookmark %s\n") % b) | ||||
repo._bookmarks[b] = repo[rb[b]].node() | ||||
write(repo) | ||||
return result | ||||
Matt Mackall
|
r11379 | def push(oldpush, ui, repo, dest=None, **opts): | ||
dopush = True | ||||
if opts.get('bookmark'): | ||||
dopush = False | ||||
for b in opts['bookmark']: | ||||
if b in repo._bookmarks: | ||||
dopush = True | ||||
opts.setdefault('rev', []).append(b) | ||||
result = 0 | ||||
if dopush: | ||||
result = oldpush(ui, repo, dest, **opts) | ||||
if opts.get('bookmark'): | ||||
# this is an unpleasant hack as push will do this internally | ||||
dest = ui.expandpath(dest or 'default-push', dest or 'default') | ||||
dest, branches = hg.parseurl(dest, opts.get('branch')) | ||||
other = hg.repository(hg.remoteui(repo, opts), dest) | ||||
rb = other.listkeys('bookmarks') | ||||
for b in opts['bookmark']: | ||||
# explicit push overrides remote bookmark if any | ||||
if b in repo._bookmarks: | ||||
ui.status(_("exporting bookmark %s\n") % b) | ||||
new = repo[b].hex() | ||||
David Soria Parra
|
r11994 | elif b in rb: | ||
Matt Mackall
|
r11379 | ui.status(_("deleting remote bookmark %s\n") % b) | ||
new = '' # delete | ||||
David Soria Parra
|
r11994 | else: | ||
Martin Geisler
|
r12146 | ui.warn(_('bookmark %s does not exist on the local ' | ||
'or remote repository!\n') % b) | ||||
David Soria Parra
|
r11994 | return 2 | ||
Matt Mackall
|
r11379 | old = rb.get(b, '') | ||
r = other.pushkey('bookmarks', b, old, new) | ||||
if not r: | ||||
ui.warn(_('updating bookmark %s failed!\n') % b) | ||||
if not result: | ||||
result = 2 | ||||
return result | ||||
David Soria Parra
|
r11431 | def diffbookmarks(ui, repo, remote): | ||
timeless
|
r12851 | ui.status(_("searching for changed bookmarks\n")) | ||
David Soria Parra
|
r11431 | |||
lmarks = repo.listkeys('bookmarks') | ||||
rmarks = remote.listkeys('bookmarks') | ||||
Patrick Mezard
|
r11682 | diff = sorted(set(rmarks) - set(lmarks)) | ||
David Soria Parra
|
r11431 | for k in diff: | ||
ui.write(" %-25s %s\n" % (k, rmarks[k][:12])) | ||||
if len(diff) <= 0: | ||||
timeless
|
r12851 | ui.status(_("no changed bookmarks found\n")) | ||
Matt Mackall
|
r11444 | return 1 | ||
return 0 | ||||
David Soria Parra
|
r11431 | |||
def incoming(oldincoming, ui, repo, source="default", **opts): | ||||
if opts.get('bookmarks'): | ||||
source, branches = hg.parseurl(ui.expandpath(source), opts.get('branch')) | ||||
other = hg.repository(hg.remoteui(repo, opts), source) | ||||
ui.status(_('comparing with %s\n') % url.hidepassword(source)) | ||||
Matt Mackall
|
r11444 | return diffbookmarks(ui, repo, other) | ||
David Soria Parra
|
r11431 | else: | ||
Matt Mackall
|
r11444 | return oldincoming(ui, repo, source, **opts) | ||
David Soria Parra
|
r11431 | |||
Matt Mackall
|
r11443 | def outgoing(oldoutgoing, ui, repo, dest=None, **opts): | ||
David Soria Parra
|
r11431 | if opts.get('bookmarks'): | ||
Matt Mackall
|
r11443 | dest = ui.expandpath(dest or 'default-push', dest or 'default') | ||
dest, branches = hg.parseurl(dest, opts.get('branch')) | ||||
other = hg.repository(hg.remoteui(repo, opts), dest) | ||||
ui.status(_('comparing with %s\n') % url.hidepassword(dest)) | ||||
Matt Mackall
|
r11444 | return diffbookmarks(ui, other, repo) | ||
David Soria Parra
|
r11431 | else: | ||
Matt Mackall
|
r11444 | return oldoutgoing(ui, repo, dest, **opts) | ||
David Soria Parra
|
r11431 | |||
Matt Mackall
|
r7638 | def uisetup(ui): | ||
extensions.wrapfunction(repair, "strip", strip) | ||||
if ui.configbool('bookmarks', 'track.current'): | ||||
extensions.wrapcommand(commands.table, 'update', updatecurbookmark) | ||||
Matt Mackall
|
r11378 | |||
entry = extensions.wrapcommand(commands.table, 'pull', pull) | ||||
entry[1].append(('B', 'bookmark', [], | ||||
Will Maier
|
r12302 | _("bookmark to import"), | ||
_('BOOKMARK'))) | ||||
Matt Mackall
|
r11379 | entry = extensions.wrapcommand(commands.table, 'push', push) | ||
entry[1].append(('B', 'bookmark', [], | ||||
Will Maier
|
r12302 | _("bookmark to export"), | ||
_('BOOKMARK'))) | ||||
David Soria Parra
|
r11431 | entry = extensions.wrapcommand(commands.table, 'incoming', incoming) | ||
entry[1].append(('B', 'bookmarks', False, | ||||
_("compare bookmark"))) | ||||
entry = extensions.wrapcommand(commands.table, 'outgoing', outgoing) | ||||
entry[1].append(('B', 'bookmarks', False, | ||||
_("compare bookmark"))) | ||||
Matt Mackall
|
r11378 | |||
Matt Mackall
|
r11372 | pushkey.register('bookmarks', pushbookmark, listbookmarks) | ||
Matt Mackall
|
r7638 | |||
David Soria Parra
|
r7481 | def updatecurbookmark(orig, ui, repo, *args, **opts): | ||
'''Set the current bookmark | ||||
If the user updates to a bookmark we update the .hg/bookmarks.current | ||||
file. | ||||
''' | ||||
res = orig(ui, repo, *args, **opts) | ||||
rev = opts['rev'] | ||||
if not rev and len(args) > 0: | ||||
rev = args[0] | ||||
setcurrent(repo, rev) | ||||
return res | ||||
Augie Fackler
|
r12714 | def bmrevset(repo, subset, x): | ||
Patrick Mezard
|
r12822 | """``bookmark([name])`` | ||
The named bookmark or all bookmarks. | ||||
""" | ||||
Martin Geisler
|
r12881 | # i18n: "bookmark" is a keyword | ||
Augie Fackler
|
r12714 | args = revset.getargs(x, 0, 1, _('bookmark takes one or no arguments')) | ||
if args: | ||||
bm = revset.getstring(args[0], | ||||
Martin Geisler
|
r12881 | # i18n: "bookmark" is a keyword | ||
Augie Fackler
|
r12714 | _('the argument to bookmark must be a string')) | ||
bmrev = listbookmarks(repo).get(bm, None) | ||||
if bmrev: | ||||
bmrev = repo.changelog.rev(bin(bmrev)) | ||||
return [r for r in subset if r == bmrev] | ||||
bms = set([repo.changelog.rev(bin(r)) for r in listbookmarks(repo).values()]) | ||||
return [r for r in subset if r in bms] | ||||
Patrick Mezard
|
r12822 | def extsetup(ui): | ||
revset.symbols['bookmark'] = bmrevset | ||||
Augie Fackler
|
r12714 | |||
David Soria Parra
|
r7239 | cmdtable = { | ||
"bookmarks": | ||||
(bookmark, | ||||
[('f', 'force', False, _('force')), | ||||
FUJIWARA Katsunori
|
r11321 | ('r', 'rev', '', _('revision'), _('REV')), | ||
David Soria Parra
|
r7239 | ('d', 'delete', False, _('delete a given bookmark')), | ||
FUJIWARA Katsunori
|
r11321 | ('m', 'rename', '', _('rename a given bookmark'), _('NAME'))], | ||
Benoit Allard
|
r7818 | _('hg bookmarks [-f] [-d] [-m NAME] [-r REV] [NAME]')), | ||
David Soria Parra
|
r7239 | } | ||
Brodie Rao
|
r10826 | |||
colortable = {'bookmarks.current': 'green'} | ||||
Patrick Mezard
|
r12823 | |||
# tell hggettext to extract docstrings from these functions: | ||||
i18nfunctions = [bmrevset] | ||||