remote-bzr: reuse bzrlib transports when possible

Pass a list of open bzrlib.transport.Transport objects to each bzrlib
function that might create a transport.  This enables bzrlib to reuse
existing transports when possible, avoiding multiple concurrent
connections to the same remote server.

If the remote server is accessed via ssh, this fixes a couple of
problems:
  * If the user does not have keys loaded into an ssh agent, the user
    may be prompted for a password multiple times.
  * If the user is using OpenSSH and the ControlMaster setting is set
    to auto, git-remote-bzr might hang.  This is because bzrlib closes
    the multiple ssh sessions in an undefined order and might try to
    close the master ssh session before the other sessions.  The
    master ssh process will not exit until the other sessions have
    exited, causing a deadlock.  (The ssh sessions are closed in an
    undefined order because bzrlib relies on the Python garbage
    collector to trigger ssh session termination.)

Signed-off-by: Richard Hansen <rhansen@bbn.com>
Acked-by: Felipe Contreras <felipe.contreras@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Richard Hansen 2013-09-08 01:47:49 -04:00 committed by Junio C Hamano
parent d2dbd399fa
commit bd5424f0d6

View File

@ -674,7 +674,7 @@ def parse_reset(parser):
parsed_refs[ref] = mark_to_rev(from_mark) parsed_refs[ref] = mark_to_rev(from_mark)
def do_export(parser): def do_export(parser):
global parsed_refs, dirname global parsed_refs, dirname, transports
parser.next() parser.next()
@ -699,7 +699,8 @@ def do_export(parser):
branch.generate_revision_history(revid, marks.get_tip(name)) branch.generate_revision_history(revid, marks.get_tip(name))
if name in peers: if name in peers:
peer = bzrlib.branch.Branch.open(peers[name]) peer = bzrlib.branch.Branch.open(peers[name],
possible_transports=transports)
try: try:
peer.bzrdir.push_branch(branch, revision_id=revid) peer.bzrdir.push_branch(branch, revision_id=revid)
except bzrlib.errors.DivergedBranches: except bzrlib.errors.DivergedBranches:
@ -769,25 +770,28 @@ def do_list(parser):
print print
def clone(path, remote_branch): def clone(path, remote_branch):
global transports
try: try:
bdir = bzrlib.bzrdir.BzrDir.create(path) bdir = bzrlib.bzrdir.BzrDir.create(path, possible_transports=transports)
except bzrlib.errors.AlreadyControlDirError: except bzrlib.errors.AlreadyControlDirError:
bdir = bzrlib.bzrdir.BzrDir.open(path) bdir = bzrlib.bzrdir.BzrDir.open(path, possible_transports=transports)
repo = bdir.find_repository() repo = bdir.find_repository()
repo.fetch(remote_branch.repository) repo.fetch(remote_branch.repository)
return remote_branch.sprout(bdir, repository=repo) return remote_branch.sprout(bdir, repository=repo)
def get_remote_branch(name): def get_remote_branch(name):
global dirname, branches global dirname, branches, transports
remote_branch = bzrlib.branch.Branch.open(branches[name]) remote_branch = bzrlib.branch.Branch.open(branches[name],
possible_transports=transports)
if isinstance(remote_branch.user_transport, bzrlib.transport.local.LocalTransport): if isinstance(remote_branch.user_transport, bzrlib.transport.local.LocalTransport):
return remote_branch return remote_branch
branch_path = os.path.join(dirname, 'clone', name) branch_path = os.path.join(dirname, 'clone', name)
try: try:
branch = bzrlib.branch.Branch.open(branch_path) branch = bzrlib.branch.Branch.open(branch_path,
possible_transports=transports)
except bzrlib.errors.NotBranchError: except bzrlib.errors.NotBranchError:
# clone # clone
branch = clone(branch_path, remote_branch) branch = clone(branch_path, remote_branch)
@ -821,17 +825,19 @@ def find_branches(repo):
yield name, branch.base yield name, branch.base
def get_repo(url, alias): def get_repo(url, alias):
global dirname, peer, branches global dirname, peer, branches, transports
normal_url = bzrlib.urlutils.normalize_url(url) normal_url = bzrlib.urlutils.normalize_url(url)
origin = bzrlib.bzrdir.BzrDir.open(url) origin = bzrlib.bzrdir.BzrDir.open(url, possible_transports=transports)
is_local = isinstance(origin.transport, bzrlib.transport.local.LocalTransport) is_local = isinstance(origin.transport, bzrlib.transport.local.LocalTransport)
shared_path = os.path.join(gitdir, 'bzr') shared_path = os.path.join(gitdir, 'bzr')
try: try:
shared_dir = bzrlib.bzrdir.BzrDir.open(shared_path) shared_dir = bzrlib.bzrdir.BzrDir.open(shared_path,
possible_transports=transports)
except bzrlib.errors.NotBranchError: except bzrlib.errors.NotBranchError:
shared_dir = bzrlib.bzrdir.BzrDir.create(shared_path) shared_dir = bzrlib.bzrdir.BzrDir.create(shared_path,
possible_transports=transports)
try: try:
shared_repo = shared_dir.open_repository() shared_repo = shared_dir.open_repository()
except bzrlib.errors.NoRepositoryPresent: except bzrlib.errors.NoRepositoryPresent:
@ -844,7 +850,8 @@ def get_repo(url, alias):
else: else:
# check and remove old organization # check and remove old organization
try: try:
bdir = bzrlib.bzrdir.BzrDir.open(clone_path) bdir = bzrlib.bzrdir.BzrDir.open(clone_path,
possible_transports=transports)
bdir.destroy_repository() bdir.destroy_repository()
except bzrlib.errors.NotBranchError: except bzrlib.errors.NotBranchError:
pass pass
@ -897,6 +904,7 @@ def main(args):
global files_cache global files_cache
global is_tmp global is_tmp
global branches, peers global branches, peers
global transports
alias = args[1] alias = args[1]
url = args[2] url = args[2]
@ -909,6 +917,7 @@ def main(args):
marks = None marks = None
branches = {} branches = {}
peers = {} peers = {}
transports = []
if alias[5:] == url: if alias[5:] == url:
is_tmp = True is_tmp = True