##// END OF EJS Templates
largefiles: wrap heads command handler more directly...
largefiles: wrap heads command handler more directly extensions.wrapfunction() is a more robust method for wrapping a function, since it allows multiple wrappers. While we're here, wrap the function registered with the command instead of installing a new command handler. Differential Revision: https://phab.mercurial-scm.org/D3178

File last commit:

r37432:2d965bfe default
r37502:c22fd3c4 default
Show More
wireproto.py
1199 lines | 44.2 KiB | text/x-python | PythonLexer
Matt Mackall
protocol: introduce wireproto.py...
r11581 # wireproto.py - generic wire protocol support functions
#
# Copyright 2005-2010 Matt Mackall <mpm@selenic.com>
#
# This software may be used and distributed according to the terms of the
# GNU General Public License version 2 or any later version.
Gregory Szorc
wireproto: use absolute_import
r25993 from __future__ import absolute_import
Augie Fackler
cleanup: replace uses of util.(md5|sha1|sha256|sha512) with hashlib.\1...
r29341 import hashlib
Gregory Szorc
wireproto: use absolute_import
r25993 import os
import tempfile
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: use absolute_import
r25993 from .i18n import _
from .node import (
bin,
hex,
Siddharth Agarwal
clone: add a server-side option to disable full getbundles (pull-based clones)...
r32260 nullid,
Gregory Szorc
wireproto: use absolute_import
r25993 )
from . import (
bundle2,
changegroup as changegroupmod,
Durham Goode
changegroup: replace changegroupsubset with makechangegroup...
r34099 discovery,
Gregory Szorc
wireproto: use absolute_import
r25993 encoding,
error,
exchange,
peer,
pushkey as pushkeymod,
Pulkit Goyal
py3: convert the mode argument of os.fdopen to unicodes (1 of 2)...
r30924 pycompat,
Gregory Szorc
wireproto: use new peer interface...
r33805 repository,
Gregory Szorc
streamclone: move code out of exchange.py...
r26443 streamclone,
Gregory Szorc
wireproto: use absolute_import
r25993 util,
Gregory Szorc
wireprototypes: move wire protocol response types to new module...
r36090 wireprototypes,
Gregory Szorc
wireproto: use absolute_import
r25993 )
Pierre-Yves David
wireproto: introduce an abstractserverproto class...
r20903
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 from .utils import (
Yuya Nishihara
procutil: bulk-replace util.std* to point to new module
r37137 procutil,
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 stringutil,
)
timeless
pycompat: switch to util.urlreq/util.urlerr for py3 compat
r28883 urlerr = util.urlerr
urlreq = util.urlreq
Pierre-Yves David
bundle1: fix bundle1-denied reporting for push over ssh...
r30909 bundle2requiredmain = _('incompatible Mercurial client; bundle2 required')
bundle2requiredhint = _('see https://www.mercurial-scm.org/wiki/'
'IncompatibleClient')
bundle2required = '%s\n(%s)\n' % (bundle2requiredmain, bundle2requiredhint)
Gregory Szorc
wireproto: config options to disable bundle1...
r27246
Augie Fackler
peer: add an iterbatcher interface...
r28436 class remoteiterbatcher(peer.iterbatcher):
def __init__(self, remote):
super(remoteiterbatcher, self).__init__()
self._remote = remote
Augie Fackler
wireproto: make iterbatcher behave streamily over http(s)...
r28438 def __getattr__(self, name):
Gregory Szorc
wireproto: properly implement batchable checking...
r33759 # Validate this method is batchable, since submit() only supports
# batchable methods.
fn = getattr(self._remote, name)
if not getattr(fn, 'batchable', None):
raise error.ProgrammingError('Attempted to batch a non-batchable '
'call to %r' % name)
Augie Fackler
wireproto: make iterbatcher behave streamily over http(s)...
r28438 return super(remoteiterbatcher, self).__getattr__(name)
Augie Fackler
peer: add an iterbatcher interface...
r28436 def submit(self):
"""Break the batch request into many patch calls and pipeline them.
This is mostly valuable over http where request sizes can be
limited, but can be used in other places as well.
"""
Gregory Szorc
wireproto: overhaul iterating batcher code (API)...
r33761 # 2-tuple of (command, arguments) that represents what will be
# sent over the wire.
requests = []
# 4-tuple of (command, final future, @batchable generator, remote
# future).
results = []
for command, args, opts, finalfuture in self.calls:
mtd = getattr(self._remote, command)
Augie Fackler
python3: replace im_{self,func} with __{self,func}__ globally...
r34727 batchable = mtd.batchable(mtd.__self__, *args, **opts)
Gregory Szorc
wireproto: overhaul iterating batcher code (API)...
r33761
commandargs, fremote = next(batchable)
assert fremote
requests.append((command, commandargs))
results.append((command, finalfuture, batchable, fremote))
if requests:
self._resultiter = self._remote._submitbatch(requests)
self._results = results
Augie Fackler
peer: add an iterbatcher interface...
r28436
def results(self):
Gregory Szorc
wireproto: overhaul iterating batcher code (API)...
r33761 for command, finalfuture, batchable, remotefuture in self._results:
# Get the raw result, set it in the remote future, feed it
# back into the @batchable generator so it can be decoded, and
# set the result on the final future to this value.
remoteresult = next(self._resultiter)
remotefuture.set(remoteresult)
finalfuture.set(next(batchable))
# Verify our @batchable generators only emit 2 values.
try:
next(batchable)
except StopIteration:
pass
else:
raise error.ProgrammingError('%s @batchable generator emitted '
'unexpected value count' % command)
yield finalfuture.value
Augie Fackler
peer: add an iterbatcher interface...
r28436
Augie Fackler
batching: migrate basic noop batching into peer.peer...
r25912 # Forward a couple of names from peer to make wireproto interactions
# slightly more sensible.
batchable = peer.batchable
future = peer.future
Peter Arrenbrecht
wireproto: add basic command batching infrastructure...
r14621
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 # list of nodes encoding / decoding
def decodelist(l, sep=' '):
Peter Arrenbrecht
wireproto: fix decodelist to properly return empty list...
r13722 if l:
Augie Fackler
wireproto: use listcomp instead of map()...
r34730 return [bin(v) for v in l.split(sep)]
Peter Arrenbrecht
wireproto: fix decodelist to properly return empty list...
r13722 return []
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597
def encodelist(l, sep=' '):
Pierre-Yves David
discovery: run discovery on filtered repository...
r23848 try:
return sep.join(map(hex, l))
except TypeError:
raise
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 # batched call argument encoding
def escapearg(plain):
return (plain
Augie Fackler
wireproto: correctly escape batched args and responses (issue4739)...
r25708 .replace(':', ':c')
.replace(',', ':o')
.replace(';', ':s')
.replace('=', ':e'))
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622
def unescapearg(escaped):
return (escaped
Augie Fackler
wireproto: correctly escape batched args and responses (issue4739)...
r25708 .replace(':e', '=')
.replace(':s', ';')
.replace(':o', ',')
.replace(':c', ':'))
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622
Gregory Szorc
wireproto: consolidate code for obtaining "cmds" argument value...
r29733 def encodebatchcmds(req):
"""Return a ``cmds`` argument value for the ``batch`` command."""
cmds = []
for op, argsdict in req:
Gregory Szorc
wireproto: unescape argument names in batch command (BC)...
r29734 # Old servers didn't properly unescape argument names. So prevent
# the sending of argument names that may not be decoded properly by
# servers.
assert all(escapearg(k) == k for k in argsdict)
Gregory Szorc
wireproto: consolidate code for obtaining "cmds" argument value...
r29733 args = ','.join('%s=%s' % (escapearg(k), escapearg(v))
for k, v in argsdict.iteritems())
cmds.append('%s %s' % (op, args))
return ';'.join(cmds)
Joerg Sonnenberger
wireproto: provide accessors for client capabilities...
r37411 def clientcompressionsupport(proto):
"""Returns a list of compression methods supported by the client.
Returns a list of the compression methods supported by the client
according to the protocol capabilities. If no such capability has
been announced, fallback to the default of zlib and uncompressed.
"""
for cap in proto.getprotocaps():
if cap.startswith('comp='):
return cap[5:].split(',')
return ['zlib', 'none']
Pierre-Yves David
getbundle: declare type of parameters...
r21646 # mapping of options accepted by getbundle and their types
#
# Meant to be extended by extensions. It is extensions responsibility to ensure
# such options are properly processed in exchange.getbundle.
#
# supported types are:
#
# :nodes: list of binary nodes
# :csv: list of comma-separated values
Pierre-Yves David
wireprotocol: distinguish list and set in getbundle argument...
r25403 # :scsv: list of comma-separated values return as set
Pierre-Yves David
getbundle: declare type of parameters...
r21646 # :plain: string with no transformation needed.
gboptsmap = {'heads': 'nodes',
Boris Feld
getbundle: add support for 'bookmarks' boolean argument...
r35268 'bookmarks': 'boolean',
Pierre-Yves David
getbundle: declare type of parameters...
r21646 'common': 'nodes',
Pierre-Yves David
getbundle: add `obsmarkers` argument to getbundle...
r22353 'obsmarkers': 'boolean',
Boris Feld
pull: use 'phase-heads' to retrieve phase information...
r34323 'phases': 'boolean',
Pierre-Yves David
wireprotocol: distinguish list and set in getbundle argument...
r25403 'bundlecaps': 'scsv',
Pierre-Yves David
getbundle: add a ``cg`` boolean argument to control changegroup inclusion...
r21989 'listkeys': 'csv',
Gregory Szorc
exchange: advertise if a clone bundle was attempted...
r26690 'cg': 'boolean',
Boris Feld
bundle2: add support for a 'stream' parameter to 'getbundle'...
r35777 'cbattempted': 'boolean',
'stream': 'boolean',
}
Pierre-Yves David
getbundle: declare type of parameters...
r21646
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 # client side
Gregory Szorc
wireproto: use new peer interface...
r33805 class wirepeer(repository.legacypeer):
Gregory Szorc
wireproto: add docstring for wirepeer
r27243 """Client-side interface for communicating with a peer repository.
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622
Gregory Szorc
wireproto: add docstring for wirepeer
r27243 Methods commonly call wire protocol commands of the same name.
See also httppeer.py and sshpeer.py for protocol-specific
implementations of this interface.
"""
Gregory Szorc
repository: port peer interfaces to zope.interface...
r37336 # Begin of ipeercommands interface.
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622
Augie Fackler
peer: add an iterbatcher interface...
r28436 def iterbatch(self):
return remoteiterbatcher(self)
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 @batchable
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 def lookup(self, key):
self.requirecap('lookup', _('look up remote revision'))
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 f = future()
Augie Fackler
wireproto: remove todict() and use {} literals instead
r20671 yield {'key': encoding.fromlocal(key)}, f
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 d = f.value
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 success, data = d[:-1].split(" ", 1)
if int(success):
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 yield bin(data)
Kyle Lippincott
wireproto: do not abort after successful lookup...
r34064 else:
self._abort(error.RepoError(data))
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 @batchable
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 def heads(self):
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 f = future()
yield {}, f
d = f.value
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 try:
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 yield decodelist(d[:-1])
Matt Mackall
wireproto: avoid naked excepts
r13726 except ValueError:
Benoit Boissinot
fix undefined variables, spotted by pylint
r11879 self._abort(error.ResponseError(_("unexpected response:"), d))
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 @batchable
Peter Arrenbrecht
wireproto: add known([id]) function...
r13723 def known(self, nodes):
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 f = future()
Augie Fackler
wireproto: remove todict() and use {} literals instead
r20671 yield {'nodes': encodelist(nodes)}, f
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 d = f.value
Peter Arrenbrecht
wireproto: add known([id]) function...
r13723 try:
Mads Kiilerich
cleanup: fix some list comprehension redefinitions of existing vars...
r22201 yield [bool(int(b)) for b in d]
Matt Mackall
wireproto: avoid naked excepts
r13726 except ValueError:
Peter Arrenbrecht
wireproto: add known([id]) function...
r13723 self._abort(error.ResponseError(_("unexpected response:"), d))
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 @batchable
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 def branchmap(self):
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 f = future()
yield {}, f
d = f.value
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 try:
branchmap = {}
for branchpart in d.splitlines():
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 branchname, branchheads = branchpart.split(' ', 1)
timeless
pycompat: switch to util.urlreq/util.urlerr for py3 compat
r28883 branchname = encoding.tolocal(urlreq.unquote(branchname))
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 branchheads = decodelist(branchheads)
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 branchmap[branchname] = branchheads
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 yield branchmap
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 except TypeError:
self._abort(error.ResponseError(_("unexpected response:"), d))
Gregory Szorc
wireproto: use new peer interface...
r33805 @batchable
def listkeys(self, namespace):
if not self.capable('pushkey'):
yield {}, None
f = future()
self.ui.debug('preparing listkeys for "%s"\n' % namespace)
yield {'namespace': encoding.fromlocal(namespace)}, f
d = f.value
self.ui.debug('received listkey for "%s": %i bytes\n'
% (namespace, len(d)))
yield pushkeymod.decodekeys(d)
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 @batchable
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 def pushkey(self, namespace, key, old, new):
if not self.capable('pushkey'):
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 yield False, None
f = future()
Pierre-Yves David
pushkey: add more verbose debug output regarding pushkey...
r17293 self.ui.debug('preparing pushkey for "%s:%s"\n' % (namespace, key))
Augie Fackler
wireproto: remove todict() and use {} literals instead
r20671 yield {'namespace': encoding.fromlocal(namespace),
'key': encoding.fromlocal(key),
'old': encoding.fromlocal(old),
'new': encoding.fromlocal(new)}, f
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 d = f.value
Pierre-Yves David
wireproto: handle other server output in pushkey...
r15652 d, output = d.split('\n', 1)
David Soria Parra
wireproto: catch possible cast error in pushkey...
r13450 try:
d = bool(int(d))
except ValueError:
raise error.ResponseError(
_('push failed (unexpected response):'), d)
Pierre-Yves David
wireproto: handle other server output in pushkey...
r15652 for l in output.splitlines(True):
self.ui.status(_('remote: '), l)
Peter Arrenbrecht
wireproto: make a number of commands batchable...
r14623 yield d
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586
Matt Mackall
protocol: unify stream_out client code
r11588 def stream_out(self):
return self._callstream('stream_out')
Pierre-Yves David
getbundle: declare type of parameters...
r21646 def getbundle(self, source, **kwargs):
Augie Fackler
wireproto: bounce kwargs to/from bytes/str as needed...
r34740 kwargs = pycompat.byteskwargs(kwargs)
Peter Arrenbrecht
wireproto: add getbundle() function...
r13741 self.requirecap('getbundle', _('look up remote changes'))
opts = {}
Joerg Sonnenberger
wireproto: don't special case bundlecaps, but sort all scsv arguments...
r37430 bundlecaps = kwargs.get('bundlecaps') or set()
Pierre-Yves David
getbundle: declare type of parameters...
r21646 for key, value in kwargs.iteritems():
if value is None:
continue
keytype = gboptsmap.get(key)
if keytype is None:
Augie Fackler
wireproto: use a proper exception instead of `assert False`...
r34731 raise error.ProgrammingError(
'Unexpectedly None keytype for key %s' % key)
Pierre-Yves David
getbundle: declare type of parameters...
r21646 elif keytype == 'nodes':
value = encodelist(value)
Joerg Sonnenberger
wireproto: don't special case bundlecaps, but sort all scsv arguments...
r37430 elif keytype == 'csv':
Pierre-Yves David
getbundle: declare type of parameters...
r21646 value = ','.join(value)
Joerg Sonnenberger
wireproto: don't special case bundlecaps, but sort all scsv arguments...
r37430 elif keytype == 'scsv':
value = ','.join(sorted(value))
Pierre-Yves David
wireproto: add a ``boolean`` type for getbundle parameters...
r21988 elif keytype == 'boolean':
Pierre-Yves David
wireprotocol: fix 'boolean' handling...
r22351 value = '%i' % bool(value)
Pierre-Yves David
getbundle: declare type of parameters...
r21646 elif keytype != 'plain':
raise KeyError('unknown getbundle option type %s'
% keytype)
opts[key] = value
Augie Fackler
wireproto: bounce kwargs to/from bytes/str as needed...
r34740 f = self._callcompressable("getbundle", **pycompat.strkwargs(opts))
Augie Fackler
cleanup: use __builtins__.any instead of util.any...
r25149 if any((cap.startswith('HG2') for cap in bundlecaps)):
Pierre-Yves David
unbundle20: retrieve unbundler instances through a factory function...
r24641 return bundle2.getunbundler(self.ui, f)
Pierre-Yves David
bundle2: allow bundle2 for pulling over the wire...
r21069 else:
Sune Foldager
changegroup: rename bundle-related functions and classes...
r22390 return changegroupmod.cg1unpacker(f, 'UN')
Peter Arrenbrecht
wireproto: add getbundle() function...
r13741
Augie Fackler
wirepeer: rename confusing `source` parameter...
r29706 def unbundle(self, cg, heads, url):
Matt Mackall
protocol: unify client unbundle support...
r11592 '''Send cg (a readable file-like object representing the
changegroup to push, typically a chunkbuffer object) to the
Pierre-Yves David
bundle2: support for push over the wire...
r21075 remote server as a bundle.
When pushing a bundle10 stream, return an integer indicating the
Martin von Zweigbergk
wireproto: update reference to deleted addchangegroup()...
r32880 result of the push (see changegroup.apply()).
Pierre-Yves David
bundle2: support for push over the wire...
r21075
Augie Fackler
wirepeer: rename confusing `source` parameter...
r29706 When pushing a bundle20 stream, return a bundle20 stream.
`url` is the url the client thinks it's pushing to, which is
visible to hooks.
'''
Matt Mackall
protocol: unify client unbundle support...
r11592
Martin Geisler
wireproto: do not hash when heads == ['force']...
r14419 if heads != ['force'] and self.capable('unbundlehash'):
Shuhei Takahashi
wireproto: allow unbundle with hashed heads parameter (issue2126)...
r13942 heads = encodelist(['hashed',
Augie Fackler
cleanup: replace uses of util.(md5|sha1|sha256|sha512) with hashlib.\1...
r29341 hashlib.sha1(''.join(sorted(heads))).digest()])
Shuhei Takahashi
wireproto: allow unbundle with hashed heads parameter (issue2126)...
r13942 else:
heads = encodelist(heads)
Pierre-Yves David
bundle2: support for push over the wire...
r21075 if util.safehasattr(cg, 'deltaheader'):
# this a bundle10, do the old style call sequence
ret, output = self._callpush("unbundle", cg, heads=heads)
if ret == "":
raise error.ResponseError(
_('push failed:'), output)
try:
ret = int(ret)
except ValueError:
raise error.ResponseError(
_('push failed (unexpected response):'), ret)
Matt Mackall
protocol: unify client unbundle support...
r11592
Pierre-Yves David
bundle2: support for push over the wire...
r21075 for l in output.splitlines(True):
self.ui.status(_('remote: '), l)
else:
# bundle2 push. Send a stream, fetch a stream.
stream = self._calltwowaystream('unbundle', cg, heads=heads)
Pierre-Yves David
unbundle20: retrieve unbundler instances through a factory function...
r24641 ret = bundle2.getunbundler(self.ui, stream)
Matt Mackall
protocol: unify client unbundle support...
r11592 return ret
Gregory Szorc
repository: port peer interfaces to zope.interface...
r37336 # End of ipeercommands interface.
Gregory Szorc
wireproto: use new peer interface...
r33805
Gregory Szorc
repository: port peer interfaces to zope.interface...
r37336 # Begin of ipeerlegacycommands interface.
Gregory Szorc
wireproto: use new peer interface...
r33805
def branches(self, nodes):
n = encodelist(nodes)
d = self._call("branches", nodes=n)
try:
br = [tuple(decodelist(b)) for b in d.splitlines()]
return br
except ValueError:
self._abort(error.ResponseError(_("unexpected response:"), d))
def between(self, pairs):
batch = 8 # avoid giant requests
r = []
for i in xrange(0, len(pairs), batch):
n = " ".join([encodelist(p, '-') for p in pairs[i:i + batch]])
d = self._call("between", pairs=n)
try:
r.extend(l and decodelist(l) or [] for l in d.splitlines())
except ValueError:
self._abort(error.ResponseError(_("unexpected response:"), d))
return r
def changegroup(self, nodes, kind):
n = encodelist(nodes)
f = self._callcompressable("changegroup", roots=n)
return changegroupmod.cg1unpacker(f, 'UN')
def changegroupsubset(self, bases, heads, kind):
self.requirecap('changegroupsubset', _('look up remote changes'))
bases = encodelist(bases)
heads = encodelist(heads)
f = self._callcompressable("changegroupsubset",
bases=bases, heads=heads)
return changegroupmod.cg1unpacker(f, 'UN')
Gregory Szorc
repository: port peer interfaces to zope.interface...
r37336 # End of ipeerlegacycommands interface.
Gregory Szorc
wireproto: use new peer interface...
r33805
def _submitbatch(self, req):
"""run batch request <req> on the server
Returns an iterator of the raw responses from the server.
"""
Boris Feld
peer-request: include more details about batch commands...
r36963 ui = self.ui
if ui.debugflag and ui.configbool('devel', 'debug.peer-request'):
ui.debug('devel-peer-request: batched-content\n')
for op, args in req:
msg = 'devel-peer-request: - %s (%d arguments)\n'
ui.debug(msg % (op, len(args)))
Gregory Szorc
wireproto: use new peer interface...
r33805 rsp = self._callstream("batch", cmds=encodebatchcmds(req))
chunk = rsp.read(1024)
work = [chunk]
while chunk:
while ';' not in chunk and chunk:
chunk = rsp.read(1024)
work.append(chunk)
merged = ''.join(work)
while ';' in merged:
one, merged = merged.split(';', 1)
yield unescapearg(one)
chunk = rsp.read(1024)
work = [merged, chunk]
yield unescapearg(''.join(work))
def _submitone(self, op, args):
Augie Fackler
wireproto: bounce kwargs to/from bytes/str as needed...
r34740 return self._call(op, **pycompat.strkwargs(args))
Gregory Szorc
wireproto: use new peer interface...
r33805
Peter Arrenbrecht
wireproto: add test for new optional arg missing on server...
r14048 def debugwireargs(self, one, two, three=None, four=None, five=None):
Peter Arrenbrecht
debug: add debugwireargs to test argument passing over the wire...
r13720 # don't pass optional arguments left at their default value
opts = {}
if three is not None:
Pulkit Goyal
py3: handle keyword arguments correctly in wireproto.py...
r35375 opts[r'three'] = three
Peter Arrenbrecht
debug: add debugwireargs to test argument passing over the wire...
r13720 if four is not None:
Pulkit Goyal
py3: handle keyword arguments correctly in wireproto.py...
r35375 opts[r'four'] = four
Peter Arrenbrecht
debug: add debugwireargs to test argument passing over the wire...
r13720 return self._call('debugwireargs', one=one, two=two, **opts)
Pierre-Yves David
wireproto: document protocol specific function of wirepeer...
r20904 def _call(self, cmd, **args):
"""execute <cmd> on the server
The command is expected to return a simple string.
returns the server reply as a string."""
raise NotImplementedError()
def _callstream(self, cmd, **args):
"""execute <cmd> on the server
Augie Fackler
wireproto: document quirk of _callstream between http and ssh...
r28435 The command is expected to return a stream. Note that if the
command doesn't return a stream, _callstream behaves
differently for ssh and http peers.
Pierre-Yves David
wireproto: document protocol specific function of wirepeer...
r20904
Augie Fackler
wireproto: document quirk of _callstream between http and ssh...
r28435 returns the server reply as a file like object.
"""
Pierre-Yves David
wireproto: document protocol specific function of wirepeer...
r20904 raise NotImplementedError()
Pierre-Yves David
wireproto: drop the _decompress method in favor a new call type...
r20905 def _callcompressable(self, cmd, **args):
"""execute <cmd> on the server
The command is expected to return a stream.
Mads Kiilerich
spelling: fixes from spell checker
r21024 The stream may have been compressed in some implementations. This
Pierre-Yves David
wireproto: drop the _decompress method in favor a new call type...
r20905 function takes care of the decompression. This is the only difference
with _callstream.
returns the server reply as a file like object.
"""
raise NotImplementedError()
Pierre-Yves David
wireproto: document protocol specific function of wirepeer...
r20904 def _callpush(self, cmd, fp, **args):
"""execute a <cmd> on server
The command is expected to be related to a push. Push has a special
return method.
returns the server reply as a (ret, output) tuple. ret is either
empty (error) or a stringified int.
"""
raise NotImplementedError()
Pierre-Yves David
wireproto: add a _calltwowaystream method on wirepeer...
r21072 def _calltwowaystream(self, cmd, fp, **args):
"""execute <cmd> on server
The command will send a stream to the server and get a stream in reply.
"""
raise NotImplementedError()
Pierre-Yves David
wireproto: document protocol specific function of wirepeer...
r20904 def _abort(self, exception):
"""clearly abort the wire protocol connection and raise the exception
"""
raise NotImplementedError()
Matt Mackall
protocol: move basic ssh client commands to wirerepository
r11586 # server side
Pierre-Yves David
wireproto: document possible return type...
r20902 # wire protocol command can either return a string or one of these classes.
Andrew Pritchard
wireproto: add out-of-band error class to allow remote repo to report errors...
r15017
Gregory Szorc
wireproto: extract repo filtering to standalone function...
r29590 def getdispatchrepo(repo, proto, command):
"""Obtain the repo used for processing wire protocol commands.
The intent of this function is to serve as a monkeypatch point for
extensions that need commands to operate on different repo views under
specialized circumstances.
"""
return repo.filtered('served')
Matt Mackall
protocol: introduce wireproto.py...
r11581 def dispatch(repo, proto, command):
Gregory Szorc
wireproto: extract repo filtering to standalone function...
r29590 repo = getdispatchrepo(repo, proto, command)
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311
transportversion = wireprototypes.TRANSPORTS[proto.name]['version']
commandtable = commandsv2 if transportversion == 2 else commands
func, spec = commandtable[command]
Matt Mackall
protocol: introduce wireproto.py...
r11581 args = proto.getargs(spec)
Dirkjan Ochtman
protocol: wrap non-string protocol responses in classes
r11625 return func(repo, proto, *args)
Matt Mackall
protocol: introduce wireproto.py...
r11581
Peter Arrenbrecht
wireproto: fix handling of '*' args for HTTP and SSH
r13721 def options(cmd, keys, others):
opts = {}
for k in keys:
if k in others:
opts[k] = others[k]
del others[k]
if others:
Yuya Nishihara
procutil: bulk-replace util.std* to point to new module
r37137 procutil.stderr.write("warning: %s ignored unexpected arguments %s\n"
% (cmd, ",".join(others)))
Peter Arrenbrecht
wireproto: fix handling of '*' args for HTTP and SSH
r13721 return opts
Gregory Szorc
wireproto: support disabling bundle1 only if repo is generaldelta...
r27633 def bundle1allowed(repo, action):
"""Whether a bundle1 operation is allowed from the server.
Priority is:
1. server.bundle1gd.<action> (if generaldelta active)
2. server.bundle1.<action>
3. server.bundle1gd (if generaldelta active)
4. server.bundle1
"""
ui = repo.ui
gd = 'generaldelta' in repo.requirements
if gd:
Boris Feld
configitems: register the 'server.bundle*' family of config...
r34614 v = ui.configbool('server', 'bundle1gd.%s' % action)
Gregory Szorc
wireproto: support disabling bundle1 only if repo is generaldelta...
r27633 if v is not None:
return v
Boris Feld
configitems: register the 'server.bundle*' family of config...
r34614 v = ui.configbool('server', 'bundle1.%s' % action)
Gregory Szorc
wireproto: config options to disable bundle1...
r27246 if v is not None:
return v
Gregory Szorc
wireproto: support disabling bundle1 only if repo is generaldelta...
r27633 if gd:
configitems: register the 'server.bundle1gd' config
r33217 v = ui.configbool('server', 'bundle1gd')
Gregory Szorc
wireproto: support disabling bundle1 only if repo is generaldelta...
r27633 if v is not None:
return v
configitems: register the 'server.bundle1' config
r33216 return ui.configbool('server', 'bundle1')
Gregory Szorc
wireproto: config options to disable bundle1...
r27246
Gregory Szorc
wireproto: remove unused proto argument from supportedcompengines (API)...
r36088 def supportedcompengines(ui, role):
Gregory Szorc
wireproto: advertise supported media types and compression formats...
r30762 """Obtain the list of supported compression engines for a request."""
assert role in (util.CLIENTROLE, util.SERVERROLE)
compengines = util.compengines.supportedwireengines(role)
# Allow config to override default list and ordering.
if role == util.SERVERROLE:
configengines = ui.configlist('server', 'compressionengines')
config = 'server.compressionengines'
else:
# This is currently implemented mainly to facilitate testing. In most
# cases, the server should be in charge of choosing a compression engine
# because a server has the most to lose from a sub-optimal choice. (e.g.
# CPU DoS due to an expensive engine or a network DoS due to poor
# compression ratio).
configengines = ui.configlist('experimental',
'clientcompressionengines')
config = 'experimental.clientcompressionengines'
# No explicit config. Filter out the ones that aren't supposed to be
# advertised and return default ordering.
if not configengines:
attr = 'serverpriority' if role == util.SERVERROLE else 'clientpriority'
return [e for e in compengines
if getattr(e.wireprotosupport(), attr) > 0]
# If compression engines are listed in the config, assume there is a good
# reason for it (like server operators wanting to achieve specific
# performance characteristics). So fail fast if the config references
# unusable compression engines.
validnames = set(e.name() for e in compengines)
invalidnames = set(e for e in configengines if e not in validnames)
if invalidnames:
raise error.Abort(_('invalid compression engine defined in %s: %s') %
(config, ', '.join(sorted(invalidnames))))
compengines = [e for e in compengines if e.name() in configengines]
compengines = sorted(compengines,
key=lambda e: configengines.index(e.name()))
if not compengines:
raise error.Abort(_('%s config option does not specify any known '
'compression engines') % config,
hint=_('usable compression engines: %s') %
', '.sorted(validnames))
return compengines
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999 class commandentry(object):
"""Represents a declared wire protocol command."""
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 def __init__(self, func, args='', transports=None,
permission='push'):
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999 self.func = func
self.args = args
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 self.transports = transports or set()
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 self.permission = permission
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999
def _merge(self, func, args):
"""Merge this instance with an incoming 2-tuple.
This is called when a caller using the old 2-tuple API attempts
to replace an instance. The incoming values are merged with
data not captured by the 2-tuple and a new instance containing
the union of the two objects is returned.
"""
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 return commandentry(func, args=args, transports=set(self.transports),
permission=self.permission)
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999
# Old code treats instances as 2-tuples. So expose that interface.
def __iter__(self):
yield self.func
yield self.args
def __getitem__(self, i):
if i == 0:
return self.func
elif i == 1:
return self.args
else:
raise IndexError('can only access elements 0 and 1')
class commanddict(dict):
"""Container for registered wire protocol commands.
It behaves like a dict. But __setitem__ is overwritten to allow silent
coercion of values from 2-tuples for API compatibility.
"""
def __setitem__(self, k, v):
if isinstance(v, commandentry):
pass
# Cast 2-tuples to commandentry instances.
elif isinstance(v, tuple):
if len(v) != 2:
raise ValueError('command tuples must have exactly 2 elements')
# It is common for extensions to wrap wire protocol commands via
# e.g. ``wireproto.commands[x] = (newfn, args)``. Because callers
# doing this aren't aware of the new API that uses objects to store
# command entries, we automatically merge old state with new.
if k in self:
v = self[k]._merge(v[0], v[1])
else:
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 # Use default values from @wireprotocommand.
v = commandentry(v[0], args=v[1],
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 transports=set(wireprototypes.TRANSPORTS),
permission='push')
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999 else:
raise ValueError('command entries must be commandentry instances '
'or 2-tuples')
return super(commanddict, self).__setitem__(k, v)
Gregory Szorc
wireproto: function for testing if wire protocol command is available...
r36000 def commandavailable(self, command, proto):
"""Determine if a command is available for the requested protocol."""
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 assert proto.name in wireprototypes.TRANSPORTS
entry = self.get(command)
if not entry:
return False
if proto.name not in entry.transports:
return False
return True
# Constants specifying which transports a wire protocol command should be
# available on. For use with @wireprotocommand.
POLICY_ALL = 'all'
POLICY_V1_ONLY = 'v1-only'
POLICY_V2_ONLY = 'v2-only'
Gregory Szorc
wireproto: function for testing if wire protocol command is available...
r36000
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 # For version 1 transports.
Gregory Szorc
wireproto: define and use types for wire protocol commands...
r35999 commands = commanddict()
Pierre-Yves David
wireproto: add decorator for wire protocol command...
r20906
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 # For version 2 transports.
commandsv2 = commanddict()
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 def wireprotocommand(name, args='', transportpolicy=POLICY_ALL,
permission='push'):
Gregory Szorc
wireproto: improve docstring for @wireprotocommand...
r35998 """Decorator to declare a wire protocol command.
``name`` is the name of the wire protocol command being provided.
``args`` is a space-delimited list of named arguments that the command
accepts. ``*`` is a special value that says to accept all arguments.
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627
``transportpolicy`` is a POLICY_* constant denoting which transports
this wire protocol command should be exposed to. By default, commands
are exposed to all wire protocol transports.
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818
``permission`` defines the permission type needed to run this command.
Can be ``push`` or ``pull``. These roughly map to read-write and read-only,
respectively. Default is to assume command requires ``push`` permissions
because otherwise commands not declaring their permissions could modify
a repository that is supposed to be read-only.
Gregory Szorc
wireproto: improve docstring for @wireprotocommand...
r35998 """
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 if transportpolicy == POLICY_ALL:
transports = set(wireprototypes.TRANSPORTS)
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 transportversions = {1, 2}
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 elif transportpolicy == POLICY_V1_ONLY:
transports = {k for k, v in wireprototypes.TRANSPORTS.items()
if v['version'] == 1}
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 transportversions = {1}
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 elif transportpolicy == POLICY_V2_ONLY:
transports = {k for k, v in wireprototypes.TRANSPORTS.items()
if v['version'] == 2}
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 transportversions = {2}
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627 else:
Gregory Szorc
wireproto: raise ProgrammingError instead of Abort...
r36858 raise error.ProgrammingError('invalid transport policy value: %s' %
transportpolicy)
Gregory Szorc
wireproto: allow wire protocol commands to declare transport support...
r36627
Gregory Szorc
wireproto: nominally don't expose "batch" to version 2 wire transports...
r37071 # Because SSHv2 is a mirror of SSHv1, we allow "batch" commands through to
# SSHv2.
# TODO undo this hack when SSH is using the unified frame protocol.
if name == b'batch':
transports.add(wireprototypes.SSHV2)
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 if permission not in ('push', 'pull'):
Gregory Szorc
wireproto: raise ProgrammingError instead of Abort...
r36858 raise error.ProgrammingError('invalid wire protocol permission; '
'got %s; expected "push" or "pull"' %
permission)
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818
Pierre-Yves David
wireproto: add decorator for wire protocol command...
r20906 def register(func):
Gregory Szorc
wireproto: separate commands tables for version 1 and 2 commands...
r37311 if 1 in transportversions:
if name in commands:
raise error.ProgrammingError('%s command already registered '
'for version 1' % name)
commands[name] = commandentry(func, args=args,
transports=transports,
permission=permission)
if 2 in transportversions:
if name in commandsv2:
raise error.ProgrammingError('%s command already registered '
'for version 2' % name)
commandsv2[name] = commandentry(func, args=args,
transports=transports,
permission=permission)
Pierre-Yves David
wireproto: add decorator for wire protocol command...
r20906 return func
return register
Gregory Szorc
wireproto: check permissions when executing "batch" command (BC) (SEC)...
r36773 # TODO define a more appropriate permissions type to use for this.
Gregory Szorc
wireproto: nominally don't expose "batch" to version 2 wire transports...
r37071 @wireprotocommand('batch', 'cmds *', permission='pull',
transportpolicy=POLICY_V1_ONLY)
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 def batch(repo, proto, cmds, others):
Kevin Bullock
filtering: rename filters to their antonyms...
r18382 repo = repo.filtered("served")
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 res = []
for pair in cmds.split(';'):
op, args = pair.split(' ', 1)
vals = {}
for a in args.split(','):
if a:
n, v = a.split('=')
Gregory Szorc
wireproto: unescape argument names in batch command (BC)...
r29734 vals[unescapearg(n)] = unescapearg(v)
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 func, spec = commands[op]
Gregory Szorc
wireproto: check permissions when executing "batch" command (BC) (SEC)...
r36773
Gregory Szorc
wireproto: formalize permissions checking as part of protocol interface...
r36819 # Validate that client has permissions to perform this command.
perm = commands[op].permission
assert perm in ('push', 'pull')
proto.checkperm(perm)
Gregory Szorc
wireproto: check permissions when executing "batch" command (BC) (SEC)...
r36773
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 if spec:
keys = spec.split()
data = {}
for k in keys:
if k == '*':
star = {}
for key in vals.keys():
if key not in keys:
star[key] = vals[key]
data['*'] = star
else:
data[k] = vals[k]
result = func(repo, proto, *[data[k] for k in keys])
else:
result = func(repo, proto)
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 if isinstance(result, wireprototypes.ooberror):
Andrew Pritchard
wireproto: add out-of-band error class to allow remote repo to report errors...
r15017 return result
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091
# For now, all batchable commands must return bytesresponse or
# raw bytes (for backwards compatibility).
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 assert isinstance(result, (wireprototypes.bytesresponse, bytes))
if isinstance(result, wireprototypes.bytesresponse):
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091 result = result.data
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622 res.append(escapearg(result))
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(';'.join(res))
Peter Arrenbrecht
wireproto: add batching support to wirerepository...
r14622
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('between', 'pairs', transportpolicy=POLICY_V1_ONLY,
permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def between(repo, proto, pairs):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 pairs = [decodelist(p, '-') for p in pairs.split(" ")]
Matt Mackall
protocol: introduce wireproto.py...
r11581 r = []
for b in repo.between(pairs):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 r.append(encodelist(b) + "\n")
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(''.join(r))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('branchmap', permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def branchmap(repo, proto):
Pierre-Yves David
clfilter: drop extra filtering in wireprotocol...
r18281 branchmap = repo.branchmap()
Matt Mackall
protocol: introduce wireproto.py...
r11581 heads = []
for branch, nodes in branchmap.iteritems():
timeless
pycompat: switch to util.urlreq/util.urlerr for py3 compat
r28883 branchname = urlreq.quote(encoding.fromlocal(branch))
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 branchnodes = encodelist(nodes)
heads.append('%s %s' % (branchname, branchnodes))
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse('\n'.join(heads))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('branches', 'nodes', transportpolicy=POLICY_V1_ONLY,
permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def branches(repo, proto, nodes):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 nodes = decodelist(nodes)
Matt Mackall
protocol: introduce wireproto.py...
r11581 r = []
for b in repo.branches(nodes):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 r.append(encodelist(b) + "\n")
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(''.join(r))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('clonebundles', '', permission='pull')
Gregory Szorc
wireproto: move clonebundles command from extension (issue4931)...
r26857 def clonebundles(repo, proto):
"""Server command for returning info for available bundles to seed clones.
Clients will parse this response and determine what bundle to fetch.
Extensions may wrap this command to filter or dynamically emit data
depending on the request. e.g. you could advertise URLs for the closest
data center given the client's IP address.
"""
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(
repo.vfs.tryread('clonebundles.manifest'))
Pierre-Yves David
wireproto: extract capabilities list in outside the wireproto function...
r20774
Gregory Szorc
wireproto: don't expose changegroupsubset capability if not available...
r36630 wireprotocaps = ['lookup', 'branchmap', 'pushkey',
Gregory Szorc
wireproto: nominally don't expose "batch" to version 2 wire transports...
r37071 'known', 'getbundle', 'unbundlehash']
Pierre-Yves David
wireproto: move wireproto capabilities computation in a subfunction...
r20775
def _capabilities(repo, proto):
"""return a list of capabilities for a repo
This function exists to allow extensions to easily wrap capabilities
computation
- returns a lists: easy to alter
- change done here will be propagated to both `capabilities` and `hello`
Mads Kiilerich
spelling: fixes from spell checker
r21024 command without any other action needed.
Pierre-Yves David
wireproto: move wireproto capabilities computation in a subfunction...
r20775 """
Pierre-Yves David
wireproto: extract capabilities list in outside the wireproto function...
r20774 # copy to prevent modification of the global list
caps = list(wireprotocaps)
Gregory Szorc
wireproto: don't expose changegroupsubset capability if not available...
r36630
# Command of same name as capability isn't exposed to version 1 of
# transports. So conditionally add it.
if commands.commandavailable('changegroupsubset', proto):
caps.append('changegroupsubset')
Gregory Szorc
streamclone: consider secret changesets (BC) (issue5589)...
r32744 if streamclone.allowservergeneration(repo):
configitems: register the 'server.preferuncompressed' config
r33222 if repo.ui.configbool('server', 'preferuncompressed'):
Benoit Allard
protocol: Add the stream-preferred capability...
r16361 caps.append('stream-preferred')
Sune Foldager
clone: only use stream when we understand the revlog format...
r12296 requiredformats = repo.requirements & repo.supportedformats
# if our local revlogs are just revlogv1, add 'stream' cap
Martin von Zweigbergk
cleanup: use set literals...
r32291 if not requiredformats - {'revlogv1'}:
Sune Foldager
clone: only use stream when we understand the revlog format...
r12296 caps.append('stream')
# otherwise, add 'streamreqs' detailing our local revlog format
else:
Pierre-Yves David
stream: sort stream capability before serialisation...
r26911 caps.append('streamreqs=%s' % ','.join(sorted(requiredformats)))
Jun Wu
codemod: register core configitems using a script...
r33499 if repo.ui.configbool('experimental', 'bundle2-advertise'):
Gregory Szorc
bundle2: specify what capabilities will be used for...
r35801 capsblob = bundle2.encodecaps(bundle2.getrepocaps(repo, role='server'))
timeless
pycompat: switch to util.urlreq/util.urlerr for py3 compat
r28883 caps.append('bundle2=' + urlreq.quote(capsblob))
Martin von Zweigbergk
bundle: move writebundle() from changegroup.py to bundle2.py (API)...
r28666 caps.append('unbundle=%s' % ','.join(bundle2.bundlepriority))
Gregory Szorc
wireproto: only advertise HTTP-specific capabilities to HTTP peers (BC)...
r30563
Gregory Szorc
wireproto: add transport specific capabilities in the transport...
r36631 return proto.addcapabilities(repo, caps)
Pierre-Yves David
wireproto: move wireproto capabilities computation in a subfunction...
r20775
Mads Kiilerich
spelling: fixes from spell checker
r21024 # If you are writing an extension and consider wrapping this function. Wrap
Pierre-Yves David
wireproto: move wireproto capabilities computation in a subfunction...
r20775 # `_capabilities` instead.
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('capabilities', permission='pull')
Pierre-Yves David
wireproto: move wireproto capabilities computation in a subfunction...
r20775 def capabilities(repo, proto):
Joerg Sonnenberger
wireproto: send server capabilities in canonical order...
r37431 caps = _capabilities(repo, proto)
return wireprototypes.bytesresponse(' '.join(sorted(caps)))
Matt Mackall
protocol: unify server-side capabilities functions
r11594
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('changegroup', 'roots', transportpolicy=POLICY_V1_ONLY,
permission='pull')
Matt Mackall
protocol: unify changegroup commands...
r11584 def changegroup(repo, proto, roots):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 nodes = decodelist(roots)
Durham Goode
changegroup: replace changegroup with makechangegroup...
r34102 outgoing = discovery.outgoing(repo, missingroots=nodes,
missingheads=repo.heads())
cg = changegroupmod.makechangegroup(repo, outgoing, '01', 'serve')
Gregory Szorc
wireproto: drop support for reader interface from streamres (API)...
r35723 gen = iter(lambda: cg.read(32768), '')
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamres(gen=gen)
Matt Mackall
protocol: unify changegroup commands...
r11584
Gregory Szorc
wireproto: don't expose legacy commands to version 2 of wire protocol...
r36629 @wireprotocommand('changegroupsubset', 'bases heads',
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 transportpolicy=POLICY_V1_ONLY,
permission='pull')
Matt Mackall
protocol: unify changegroup commands...
r11584 def changegroupsubset(repo, proto, bases, heads):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 bases = decodelist(bases)
heads = decodelist(heads)
Durham Goode
changegroup: replace changegroupsubset with makechangegroup...
r34099 outgoing = discovery.outgoing(repo, missingroots=bases,
missingheads=heads)
cg = changegroupmod.makechangegroup(repo, outgoing, '01', 'serve')
Gregory Szorc
wireproto: drop support for reader interface from streamres (API)...
r35723 gen = iter(lambda: cg.read(32768), '')
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamres(gen=gen)
Matt Mackall
protocol: unify changegroup commands...
r11584
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('debugwireargs', 'one two *',
permission='pull')
Peter Arrenbrecht
wireproto: fix handling of '*' args for HTTP and SSH
r13721 def debugwireargs(repo, proto, one, two, others):
# only accept optional args from the known set
opts = options('debugwireargs', ['three', 'four'], others)
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(repo.debugwireargs(
one, two, **pycompat.strkwargs(opts)))
Peter Arrenbrecht
debug: add debugwireargs to test argument passing over the wire...
r13720
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('getbundle', '*', permission='pull')
Peter Arrenbrecht
wireproto: add getbundle() function...
r13741 def getbundle(repo, proto, others):
Pierre-Yves David
getbundle: declare type of parameters...
r21646 opts = options('getbundle', gboptsmap.keys(), others)
Peter Arrenbrecht
wireproto: add getbundle() function...
r13741 for k, v in opts.iteritems():
Pierre-Yves David
getbundle: declare type of parameters...
r21646 keytype = gboptsmap[k]
if keytype == 'nodes':
Benoit Boissinot
bundle-ng: add bundlecaps argument to getbundle() command
r19201 opts[k] = decodelist(v)
Pierre-Yves David
getbundle: declare type of parameters...
r21646 elif keytype == 'csv':
Pierre-Yves David
wireprotocol: distinguish list and set in getbundle argument...
r25403 opts[k] = list(v.split(','))
elif keytype == 'scsv':
Benoit Boissinot
bundle-ng: add bundlecaps argument to getbundle() command
r19201 opts[k] = set(v.split(','))
Pierre-Yves David
wireproto: add a ``boolean`` type for getbundle parameters...
r21988 elif keytype == 'boolean':
Gregory Szorc
wireproto: properly parse false boolean args (BC)...
r26686 # Client should serialize False as '0', which is a non-empty string
# so it evaluates as a True bool.
if v == '0':
opts[k] = False
else:
opts[k] = bool(v)
Pierre-Yves David
getbundle: declare type of parameters...
r21646 elif keytype != 'plain':
raise KeyError('unknown getbundle option type %s'
% keytype)
Gregory Szorc
wireproto: config options to disable bundle1...
r27246
Gregory Szorc
wireproto: support disabling bundle1 only if repo is generaldelta...
r27633 if not bundle1allowed(repo, 'pull'):
Gregory Szorc
wireproto: config options to disable bundle1...
r27246 if not exchange.bundle2requested(opts.get('bundlecaps')):
Gregory Szorc
wireprotoserver: add version to HTTP protocol name (API)...
r36241 if proto.name == 'http-v1':
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.ooberror(bundle2required)
Pierre-Yves David
bundle1: fix bundle1-denied reporting for pull over ssh...
r30912 raise error.Abort(bundle2requiredmain,
hint=bundle2requiredhint)
Gregory Szorc
wireproto: config options to disable bundle1...
r27246
Gregory Szorc
exchange: send bundle2 stream clones uncompressed...
r35805 prefercompressed = True
Gregory Szorc
wireproto: don't compress errors from getbundle()...
r35800
Pierre-Yves David
wireproto: properly report server Abort during 'getbundle'...
r30914 try:
configitems: register the 'server.disablefullbundle' config
r33220 if repo.ui.configbool('server', 'disablefullbundle'):
Siddharth Agarwal
clone: add a server-side option to disable full getbundles (pull-based clones)...
r32260 # Check to see if this is a full clone.
clheads = set(repo.changelog.heads())
Boris Feld
clone: allow bundle2's stream clone with 'server.disablefullbundle'...
r35778 changegroup = opts.get('cg', True)
Siddharth Agarwal
clone: add a server-side option to disable full getbundles (pull-based clones)...
r32260 heads = set(opts.get('heads', set()))
common = set(opts.get('common', set()))
common.discard(nullid)
Boris Feld
clone: allow bundle2's stream clone with 'server.disablefullbundle'...
r35778 if changegroup and not common and clheads == heads:
Siddharth Agarwal
clone: add a server-side option to disable full getbundles (pull-based clones)...
r32260 raise error.Abort(
_('server has pull-based clones disabled'),
hint=_('remove --pull if specified or upgrade Mercurial'))
Gregory Szorc
exchange: return bundle info from getbundlechunks() (API)...
r35803 info, chunks = exchange.getbundlechunks(repo, 'serve',
**pycompat.strkwargs(opts))
Gregory Szorc
exchange: send bundle2 stream clones uncompressed...
r35805 prefercompressed = info.get('prefercompressed', True)
Pierre-Yves David
wireproto: properly report server Abort during 'getbundle'...
r30914 except error.Abort as exc:
# cleanly forward Abort error to the client
if not exchange.bundle2requested(opts.get('bundlecaps')):
Gregory Szorc
wireprotoserver: add version to HTTP protocol name (API)...
r36241 if proto.name == 'http-v1':
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.ooberror(pycompat.bytestr(exc) + '\n')
Pierre-Yves David
wireproto: properly report server Abort during 'getbundle'...
r30914 raise # cannot do better for bundle1 + ssh
# bundle2 request expect a bundle2 reply
bundler = bundle2.bundle20(repo.ui)
Augie Fackler
py3: get bytes-repr of network errors portably...
r36272 manargs = [('message', pycompat.bytestr(exc))]
Pierre-Yves David
wireproto: properly report server Abort during 'getbundle'...
r30914 advargs = []
if exc.hint is not None:
advargs.append(('hint', exc.hint))
bundler.addpart(bundle2.bundlepart('error:abort',
manargs, advargs))
Gregory Szorc
wireproto: don't compress errors from getbundle()...
r35800 chunks = bundler.getchunks()
Gregory Szorc
exchange: send bundle2 stream clones uncompressed...
r35805 prefercompressed = False
Gregory Szorc
wireproto: don't compress errors from getbundle()...
r35800
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamres(
gen=chunks, prefer_uncompressed=not prefercompressed)
Peter Arrenbrecht
wireproto: add getbundle() function...
r13741
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('heads', permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def heads(repo, proto):
Pierre-Yves David
clfilter: drop extra filtering in wireprotocol...
r18281 h = repo.heads()
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(encodelist(h) + '\n')
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('hello', permission='pull')
Matt Mackall
protocol: unify server-side capabilities functions
r11594 def hello(repo, proto):
Gregory Szorc
wireproto: improve docstring for "hello"...
r36239 """Called as part of SSH handshake to obtain server info.
Returns a list of lines describing interesting things about the
server, in an RFC822-like format.
Matt Mackall
protocol: unify server-side capabilities functions
r11594
Gregory Szorc
wireproto: improve docstring for "hello"...
r36239 Currently, the only one defined is ``capabilities``, which consists of a
line of space separated tokens describing server abilities:
capabilities: <token0> <token1> <token2>
"""
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091 caps = capabilities(repo, proto).data
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse('capabilities: %s\n' % caps)
Matt Mackall
protocol: unify server-side capabilities functions
r11594
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('listkeys', 'namespace', permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def listkeys(repo, proto, namespace):
Gregory Szorc
wireproto: sort response to listkeys...
r36546 d = sorted(repo.listkeys(encoding.tolocal(namespace)).items())
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(pushkeymod.encodekeys(d))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('lookup', 'key', permission='pull')
Matt Mackall
protocol: add proto to method prototypes
r11583 def lookup(repo, proto, key):
Matt Mackall
protocol: introduce wireproto.py...
r11581 try:
Matt Mackall
wireproto: refuse to lookup secret csets
r15925 k = encoding.tolocal(key)
Martin von Zweigbergk
wireproto: use repo.lookup() for lookup command...
r37371 n = repo.lookup(k)
r = hex(n)
Matt Mackall
protocol: introduce wireproto.py...
r11581 success = 1
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except Exception as inst:
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 r = stringutil.forcebytestr(inst)
Matt Mackall
protocol: introduce wireproto.py...
r11581 success = 0
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse('%d %s\n' % (success, r))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('known', 'nodes *', permission='pull')
Peter Arrenbrecht
wireproto: enable optional args for known() for future extensibility...
r14436 def known(repo, proto, nodes, others):
Gregory Szorc
wireproto: introduce type for raw byte responses (API)...
r36091 v = ''.join(b and '1' or '0' for b in repo.known(decodelist(nodes)))
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse(v)
Peter Arrenbrecht
wireproto: add known([id]) function...
r13723
Joerg Sonnenberger
wireproto: provide accessors for client capabilities...
r37411 @wireprotocommand('protocaps', 'caps', permission='pull',
transportpolicy=POLICY_V1_ONLY)
def protocaps(repo, proto, caps):
if proto.name == wireprototypes.SSHV1:
proto._protocaps = set(caps.split(' '))
return wireprototypes.bytesresponse('OK')
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('pushkey', 'namespace key old new', permission='push')
Matt Mackall
protocol: add proto to method prototypes
r11583 def pushkey(repo, proto, namespace, key, old, new):
Matt Mackall
pushkey: use UTF-8
r13050 # compatibility with pre-1.8 clients which were accidentally
# sending raw binary nodes rather than utf-8-encoded hex
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 if len(new) == 20 and stringutil.escapestr(new) != new:
Matt Mackall
pushkey: use UTF-8
r13050 # looks like it could be a binary node
try:
Alexander Solovyov
remove unused imports and variables
r14064 new.decode('utf-8')
Matt Mackall
pushkey: use UTF-8
r13050 new = encoding.tolocal(new) # but cleanly decodes as UTF-8
except UnicodeDecodeError:
pass # binary, leave unmodified
else:
new = encoding.tolocal(new) # normal path
Gregory Szorc
wireprotoserver: add context manager mechanism for redirecting stdio...
r36083 with proto.mayberedirectstdio() as output:
Gregory Szorc
wireproto: remove unnecessary exception trapping...
r35997 r = repo.pushkey(encoding.tolocal(namespace), encoding.tolocal(key),
encoding.tolocal(old), new) or False
Wagner Bruna
wireproto: fix pushkey hook failure and output on remote http repo...
r17793
Gregory Szorc
wireprotoserver: add context manager mechanism for redirecting stdio...
r36083 output = output.getvalue() if output else ''
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.bytesresponse('%d\n%s' % (int(r), output))
Matt Mackall
protocol: introduce wireproto.py...
r11581
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('stream_out', permission='pull')
Matt Mackall
protocol: unify stream_out command
r11585 def stream(repo, proto):
Dirkjan Ochtman
protocol: move the streamclone implementation into wireproto
r11627 '''If the server supports streaming clone, it advertises the "stream"
capability with a value representing the version and flags of the repo
it is serving. Client checks to see if it understands the format.
'''
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamreslegacy(
streamclone.generatev1wireproto(repo))
Matt Mackall
protocol: unify stream_out command
r11585
Gregory Szorc
wireproto: declare permissions requirements in @wireprotocommand (API)...
r36818 @wireprotocommand('unbundle', 'heads', permission='push')
Matt Mackall
protocol: unify unbundle on the server side
r11593 def unbundle(repo, proto, heads):
Benoit Boissinot
wireproto: refactor list of nodeid encoding / decoding
r11597 their_heads = decodelist(heads)
Matt Mackall
protocol: unify unbundle on the server side
r11593
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 with proto.mayberedirectstdio() as output:
Matt Mackall
protocol: unify unbundle on the server side
r11593 try:
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 exchange.check_heads(repo, their_heads, 'preparing changes')
Joerg Sonnenberger
wireproto: allow direct stream processing for unbundle...
r37432 cleanup = lambda: None
try:
payload = proto.getpayload()
if repo.ui.configbool('server', 'streamunbundle'):
def cleanup():
# Ensure that the full payload is consumed, so
# that the connection doesn't contain trailing garbage.
for p in payload:
pass
fp = util.chunkbuffer(payload)
else:
# write bundle data to temporary file as it can be big
fp, tempname = None, None
def cleanup():
if fp:
fp.close()
if tempname:
os.unlink(tempname)
fd, tempname = tempfile.mkstemp(prefix='hg-unbundle-')
repo.ui.debug('redirecting incoming bundle to %s\n' %
tempname)
fp = os.fdopen(fd, pycompat.sysstr('wb+'))
r = 0
for p in payload:
fp.write(p)
fp.seek(0)
Gregory Szorc
wireproto: config options to disable bundle1...
r27246
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 gen = exchange.readbundle(repo.ui, fp, None)
if (isinstance(gen, changegroupmod.cg1unpacker)
and not bundle1allowed(repo, 'push')):
Gregory Szorc
wireprotoserver: add version to HTTP protocol name (API)...
r36241 if proto.name == 'http-v1':
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 # need to special case http because stderr do not get to
# the http client on failed push so we need to abuse
# some other error type to make sure the message get to
# the user.
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.ooberror(bundle2required)
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 raise error.Abort(bundle2requiredmain,
hint=bundle2requiredhint)
Pierre-Yves David
bundle2: refactor error bundle creation for the wireprotocol...
r24796
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 r = exchange.unbundle(repo, gen, their_heads, 'serve',
Gregory Szorc
wireprotoserver: rename _client to client (API)...
r36086 proto.client())
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 if util.safehasattr(r, 'addpart'):
# The return looks streamable, we are in the bundle2 case
# and should return a stream.
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamreslegacy(gen=r.getchunks())
return wireprototypes.pushres(
r, output.getvalue() if output else '')
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084
finally:
Joerg Sonnenberger
wireproto: allow direct stream processing for unbundle...
r37432 cleanup()
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084
except (error.BundleValueError, error.Abort, error.PushRaced) as exc:
# handle non-bundle2 case first
if not getattr(exc, 'duringunbundle2', False):
try:
Pierre-Yves David
bundle2: convey PushkeyFailed error over the wire...
r25493 raise
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 except error.Abort:
Yuya Nishihara
procutil: bulk-replace util.std* to point to new module
r37137 # The old code we moved used procutil.stderr directly.
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 # We did not change it to minimise code change.
# This need to be moved to something proper.
# Feel free to do it.
Yuya Nishihara
procutil: bulk-replace util.std* to point to new module
r37137 procutil.stderr.write("abort: %s\n" % exc)
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 if exc.hint is not None:
Yuya Nishihara
procutil: bulk-replace util.std* to point to new module
r37137 procutil.stderr.write("(%s)\n" % exc.hint)
procutil.stderr.flush()
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.pushres(
0, output.getvalue() if output else '')
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 except error.PushRaced:
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.pusherr(
pycompat.bytestr(exc),
output.getvalue() if output else '')
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084
bundler = bundle2.bundle20(repo.ui)
for out in getattr(exc, '_bundle2salvagedoutput', ()):
bundler.addpart(out)
try:
try:
raise
except error.PushkeyFailed as exc:
# check client caps
remotecaps = getattr(exc, '_replycaps', None)
if (remotecaps is not None
and 'pushkey' not in remotecaps.get('error', ())):
# no support remote side, fallback to Abort handler.
raise
part = bundler.newpart('error:pushkey')
part.addparam('in-reply-to', exc.partid)
if exc.namespace is not None:
part.addparam('namespace', exc.namespace,
mandatory=False)
if exc.key is not None:
part.addparam('key', exc.key, mandatory=False)
if exc.new is not None:
part.addparam('new', exc.new, mandatory=False)
if exc.old is not None:
part.addparam('old', exc.old, mandatory=False)
if exc.ret is not None:
part.addparam('ret', exc.ret, mandatory=False)
except error.BundleValueError as exc:
errpart = bundler.newpart('error:unsupportedcontent')
if exc.parttype is not None:
errpart.addparam('parttype', exc.parttype)
if exc.params:
errpart.addparam('params', '\0'.join(exc.params))
except error.Abort as exc:
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 manargs = [('message', stringutil.forcebytestr(exc))]
Gregory Szorc
wireproto: use maybecapturestdio() for push responses (API)...
r36084 advargs = []
if exc.hint is not None:
advargs.append(('hint', exc.hint))
bundler.addpart(bundle2.bundlepart('error:abort',
manargs, advargs))
except error.PushRaced as exc:
Augie Fackler
wireproto: fix lingering str(exception) with util.forcebytestr(exception)...
r36332 bundler.newpart('error:pushraced',
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 [('message', stringutil.forcebytestr(exc))])
Gregory Szorc
wireproto: stop aliasing wire protocol types (API)...
r37309 return wireprototypes.streamreslegacy(gen=bundler.getchunks())