Show More
share.py
212 lines
| 7.0 KiB
| text/x-python
|
PythonLexer
/ hgext / share.py
Matt Mackall
|
r8801 | # Copyright 2006, 2007 Matt Mackall <mpm@selenic.com> | ||
# | ||||
# 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. | ||
Matt Mackall
|
r8801 | |||
Gregory Szorc
|
r25761 | '''share a common history between several working directories | ||
Joerg Sonnenberger
|
r45535 | The share extension introduces a new command :hg:`share` to create a new | ||
working directory. This is similar to :hg:`clone`, but doesn't involve | ||||
copying or linking the storage of the repository. This allows working on | ||||
different branches or changes in parallel without the associated cost in | ||||
terms of disk space. | ||||
Note: destructive operations or extensions like :hg:`rollback` should be | ||||
used with care as they can result in confusing problems. | ||||
Gregory Szorc
|
r25761 | Automatic Pooled Storage for Clones | ||
----------------------------------- | ||||
When this extension is active, :hg:`clone` can be configured to | ||||
automatically share/pool storage across multiple clones. This | ||||
mode effectively converts :hg:`clone` to :hg:`clone` + :hg:`share`. | ||||
The benefit of using this mode is the automatic management of | ||||
store paths and intelligent pooling of related repositories. | ||||
The following ``share.`` config options influence this feature: | ||||
Matt Mackall
|
r25851 | ``share.pool`` | ||
Gregory Szorc
|
r25761 | Filesystem path where shared repository data will be stored. When | ||
defined, :hg:`clone` will automatically use shared repository | ||||
storage instead of creating a store inside each clone. | ||||
Matt Mackall
|
r25851 | ``share.poolnaming`` | ||
Gregory Szorc
|
r25761 | How directory names in ``share.pool`` are constructed. | ||
"identity" means the name is derived from the first changeset in the | ||||
repository. In this mode, different remotes share storage if their | ||||
root/initial changeset is identical. In this mode, the local shared | ||||
repository is an aggregate of all encountered remote repositories. | ||||
"remote" means the name is derived from the source repository's | ||||
path or URL. In this mode, storage is only shared if the path or URL | ||||
requested in the :hg:`clone` command matches exactly to a repository | ||||
that was cloned before. | ||||
Matt Harbison
|
r34950 | The default naming mode is "identity". | ||
Gregory Szorc
|
r25761 | ''' | ||
Dirkjan Ochtman
|
r8873 | |||
Pulkit Goyal
|
r29485 | from __future__ import absolute_import | ||
import errno | ||||
Matt Mackall
|
r8801 | from mercurial.i18n import _ | ||
Pulkit Goyal
|
r29485 | from mercurial import ( | ||
bookmarks, | ||||
commands, | ||||
error, | ||||
extensions, | ||||
hg, | ||||
Yuya Nishihara
|
r32337 | registrar, | ||
FUJIWARA Katsunori
|
r31052 | txnutil, | ||
Pulkit Goyal
|
r29485 | util, | ||
) | ||||
Gregory Szorc
|
r21253 | cmdtable = {} | ||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Augie Fackler
|
r16743 | |||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'share', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | (b'U', b'noupdate', None, _(b'do not create a working directory')), | ||
(b'B', b'bookmarks', None, _(b'also share bookmarks')), | ||||
Pulkit Goyal
|
r43564 | (b'', b'relative', None, _(b'point to source using a relative path'),), | ||
Dan Villiom Podlaski Christiansen
|
r31133 | ], | ||
Augie Fackler
|
r43347 | _(b'[-U] [-B] SOURCE [DEST]'), | ||
rdamazio@google.com
|
r40329 | helpcategory=command.CATEGORY_REPO_CREATION, | ||
Augie Fackler
|
r43346 | norepo=True, | ||
) | ||||
def share( | ||||
ui, source, dest=None, noupdate=False, bookmarks=False, relative=False | ||||
): | ||||
Martin Geisler
|
r10798 | """create a new shared repository | ||
Matt Mackall
|
r8801 | |||
Martin Geisler
|
r9273 | Initialize a new repository and working directory that shares its | ||
Ryan McElroy
|
r23614 | history (and optionally bookmarks) with another repository. | ||
Matt Mackall
|
r8801 | |||
Erik Zielke
|
r12389 | .. note:: | ||
Simon Heimberg
|
r19997 | |||
Erik Zielke
|
r12389 | using rollback or extensions that destroy/modify history (mq, | ||
rebase, etc.) can cause considerable confusion with shared | ||||
clones. In particular, if two shared clones are both updated to | ||||
the same changeset, and one of them destroys that changeset | ||||
with rollback, the other clone will suddenly stop working: all | ||||
operations will fail with "abort: working directory has unknown | ||||
parent". The only known workaround is to use debugsetparents on | ||||
Matt Mackall
|
r19399 | the broken clone to reset it to a changeset that still exists. | ||
Matt Mackall
|
r8801 | """ | ||
Augie Fackler
|
r43346 | hg.share( | ||
ui, | ||||
source, | ||||
dest=dest, | ||||
update=not noupdate, | ||||
bookmarks=bookmarks, | ||||
relative=relative, | ||||
) | ||||
Matt Harbison
|
r34816 | return 0 | ||
Matt Mackall
|
r8801 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @command(b'unshare', [], b'', helpcategory=command.CATEGORY_MAINTENANCE) | ||
Simon Heimberg
|
r15079 | def unshare(ui, repo): | ||
"""convert a shared repository to a normal one | ||||
Copy the store data to the repo and remove the sharedpath data. | ||||
""" | ||||
Angel Ezquerra
|
r23666 | if not repo.shared(): | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"this is not a shared repo")) | ||
Simon Heimberg
|
r15079 | |||
Matt Harbison
|
r34879 | hg.unshare(ui, repo) | ||
Ryan McElroy
|
r23548 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r25761 | # Wrap clone command to pass auto share options. | ||
def clone(orig, ui, source, *args, **opts): | ||||
Augie Fackler
|
r43347 | pool = ui.config(b'share', b'pool') | ||
Gregory Szorc
|
r25761 | if pool: | ||
pool = util.expandpath(pool) | ||||
Augie Fackler
|
r43906 | opts['shareopts'] = { | ||
Augie Fackler
|
r43347 | b'pool': pool, | ||
b'mode': ui.config(b'share', b'poolnaming'), | ||||
Yuya Nishihara
|
r33021 | } | ||
Gregory Szorc
|
r25761 | |||
return orig(ui, source, *args, **opts) | ||||
Augie Fackler
|
r43346 | |||
Ryan McElroy
|
r23548 | def extsetup(ui): | ||
Augie Fackler
|
r43347 | extensions.wrapfunction(bookmarks, b'_getbkfile', getbkfile) | ||
extensions.wrapfunction(bookmarks.bmstore, b'_recordchange', recordchange) | ||||
extensions.wrapfunction(bookmarks.bmstore, b'_writerepo', writerepo) | ||||
extensions.wrapcommand(commands.table, b'clone', clone) | ||||
Ryan McElroy
|
r23548 | |||
Augie Fackler
|
r43346 | |||
Ryan McElroy
|
r23548 | def _hassharedbookmarks(repo): | ||
"""Returns whether this repo has shared bookmarks""" | ||||
Martin von Zweigbergk
|
r42512 | if bookmarks.bookmarksinstore(repo): | ||
# Kind of a lie, but it means that we skip our custom reads and writes | ||||
# from/to the source repo. | ||||
return False | ||||
Ryan McElroy
|
r23548 | try: | ||
Augie Fackler
|
r43347 | shared = repo.vfs.read(b'shared').splitlines() | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Ryan McElroy
|
r23548 | if inst.errno != errno.ENOENT: | ||
raise | ||||
return False | ||||
Martijn Pieters
|
r29424 | return hg.sharedbookmarks in shared | ||
Ryan McElroy
|
r23548 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r27186 | def getbkfile(orig, repo): | ||
Ryan McElroy
|
r23548 | if _hassharedbookmarks(repo): | ||
Gregory Szorc
|
r36177 | srcrepo = hg.sharedreposource(repo) | ||
Ryan McElroy
|
r23548 | if srcrepo is not None: | ||
FUJIWARA Katsunori
|
r31052 | # just orig(srcrepo) doesn't work as expected, because | ||
# HG_PENDING refers repo.root. | ||||
try: | ||||
Augie Fackler
|
r43346 | fp, pending = txnutil.trypending( | ||
Augie Fackler
|
r43347 | repo.root, repo.vfs, b'bookmarks' | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r31052 | if pending: | ||
# only in this case, bookmark information in repo | ||||
# is up-to-date. | ||||
return fp | ||||
fp.close() | ||||
except IOError as inst: | ||||
if inst.errno != errno.ENOENT: | ||||
raise | ||||
# otherwise, we should read bookmarks from srcrepo, | ||||
# because .hg/bookmarks in srcrepo might be already | ||||
# changed via another sharing repo | ||||
Ryan McElroy
|
r23548 | repo = srcrepo | ||
FUJIWARA Katsunori
|
r31052 | |||
# TODO: Pending changes in repo are still invisible in | ||||
# srcrepo, because bookmarks.pending is written only into repo. | ||||
# See also https://www.mercurial-scm.org/wiki/SharedRepository | ||||
Augie Fackler
|
r27186 | return orig(repo) | ||
Ryan McElroy
|
r23548 | |||
Augie Fackler
|
r43346 | |||
Ryan McElroy
|
r23548 | def recordchange(orig, self, tr): | ||
# Continue with write to local bookmarks file as usual | ||||
orig(self, tr) | ||||
if _hassharedbookmarks(self._repo): | ||||
Gregory Szorc
|
r36177 | srcrepo = hg.sharedreposource(self._repo) | ||
Ryan McElroy
|
r23548 | if srcrepo is not None: | ||
Augie Fackler
|
r43347 | category = b'share-bookmarks' | ||
Ryan McElroy
|
r23548 | tr.addpostclose(category, lambda tr: self._writerepo(srcrepo)) | ||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r26933 | def writerepo(orig, self, repo): | ||
Ryan McElroy
|
r23548 | # First write local bookmarks file in case we ever unshare | ||
FUJIWARA Katsunori
|
r26933 | orig(self, repo) | ||
Ryan McElroy
|
r23548 | if _hassharedbookmarks(self._repo): | ||
Gregory Szorc
|
r36177 | srcrepo = hg.sharedreposource(self._repo) | ||
Ryan McElroy
|
r23548 | if srcrepo is not None: | ||
FUJIWARA Katsunori
|
r26933 | orig(self, srcrepo) | ||