wireprototypes.py
156 lines
| 4.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / wireprototypes.py
Gregory Szorc
|
r36090 | # Copyright 2018 Gregory Szorc <gregory.szorc@gmail.com> | ||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
Gregory Szorc
|
r37312 | from .thirdparty.zope import ( | ||
interface as zi, | ||||
) | ||||
Gregory Szorc
|
r36389 | |||
Gregory Szorc
|
r36553 | # Names of the SSH protocol implementations. | ||
SSHV1 = 'ssh-v1' | ||||
Gregory Szorc
|
r37064 | # These are advertised over the wire. Increment the counters at the end | ||
Gregory Szorc
|
r36553 | # to reflect BC breakages. | ||
SSHV2 = 'exp-ssh-v2-0001' | ||||
Gregory Szorc
|
r37064 | HTTPV2 = 'exp-http-v2-0001' | ||
Gregory Szorc
|
r36553 | |||
Gregory Szorc
|
r36627 | # All available wire protocol transports. | ||
TRANSPORTS = { | ||||
SSHV1: { | ||||
'transport': 'ssh', | ||||
'version': 1, | ||||
}, | ||||
SSHV2: { | ||||
'transport': 'ssh', | ||||
Gregory Szorc
|
r37310 | # TODO mark as version 2 once all commands are implemented. | ||
'version': 1, | ||||
Gregory Szorc
|
r36627 | }, | ||
'http-v1': { | ||||
'transport': 'http', | ||||
'version': 1, | ||||
Gregory Szorc
|
r37064 | }, | ||
HTTPV2: { | ||||
'transport': 'http', | ||||
'version': 2, | ||||
Gregory Szorc
|
r36627 | } | ||
} | ||||
Gregory Szorc
|
r36091 | class bytesresponse(object): | ||
"""A wire protocol response consisting of raw bytes.""" | ||||
def __init__(self, data): | ||||
self.data = data | ||||
Gregory Szorc
|
r36090 | class ooberror(object): | ||
"""wireproto reply: failure of a batch of operation | ||||
Something failed during a batch call. The error message is stored in | ||||
`self.message`. | ||||
""" | ||||
def __init__(self, message): | ||||
self.message = message | ||||
class pushres(object): | ||||
"""wireproto reply: success with simple integer return | ||||
The call was successful and returned an integer contained in `self.res`. | ||||
""" | ||||
def __init__(self, res, output): | ||||
self.res = res | ||||
self.output = output | ||||
class pusherr(object): | ||||
"""wireproto reply: failure | ||||
The call failed. The `self.res` attribute contains the error message. | ||||
""" | ||||
def __init__(self, res, output): | ||||
self.res = res | ||||
self.output = output | ||||
class streamres(object): | ||||
"""wireproto reply: binary stream | ||||
The call was successful and the result is a stream. | ||||
Accepts a generator containing chunks of data to be sent to the client. | ||||
``prefer_uncompressed`` indicates that the data is expected to be | ||||
uncompressable and that the stream should therefore use the ``none`` | ||||
engine. | ||||
""" | ||||
def __init__(self, gen=None, prefer_uncompressed=False): | ||||
self.gen = gen | ||||
self.prefer_uncompressed = prefer_uncompressed | ||||
class streamreslegacy(object): | ||||
"""wireproto reply: uncompressed binary stream | ||||
The call was successful and the result is a stream. | ||||
Accepts a generator containing chunks of data to be sent to the client. | ||||
Like ``streamres``, but sends an uncompressed data for "version 1" clients | ||||
using the application/mercurial-0.1 media type. | ||||
""" | ||||
def __init__(self, gen=None): | ||||
self.gen = gen | ||||
Gregory Szorc
|
r36389 | |||
Gregory Szorc
|
r37312 | class baseprotocolhandler(zi.Interface): | ||
Gregory Szorc
|
r36389 | """Abstract base class for wire protocol handlers. | ||
A wire protocol handler serves as an interface between protocol command | ||||
handlers and the wire protocol transport layer. Protocol handlers provide | ||||
methods to read command arguments, redirect stdio for the duration of | ||||
the request, handle response types, etc. | ||||
""" | ||||
Gregory Szorc
|
r37312 | name = zi.Attribute( | ||
Gregory Szorc
|
r36389 | """The name of the protocol implementation. | ||
Used for uniquely identifying the transport type. | ||||
Gregory Szorc
|
r37312 | """) | ||
Gregory Szorc
|
r36389 | |||
Gregory Szorc
|
r37312 | def getargs(args): | ||
Gregory Szorc
|
r36389 | """return the value for arguments in <args> | ||
returns a list of values (same order as <args>)""" | ||||
Gregory Szorc
|
r37312 | def forwardpayload(fp): | ||
Gregory Szorc
|
r36389 | """Read the raw payload and forward to a file. | ||
The payload is read in full before the function returns. | ||||
""" | ||||
Gregory Szorc
|
r37312 | def mayberedirectstdio(): | ||
Gregory Szorc
|
r36389 | """Context manager to possibly redirect stdio. | ||
The context manager yields a file-object like object that receives | ||||
stdout and stderr output when the context manager is active. Or it | ||||
yields ``None`` if no I/O redirection occurs. | ||||
The intent of this context manager is to capture stdio output | ||||
so it may be sent in the response. Some transports support streaming | ||||
stdio to the client in real time. For these transports, stdio output | ||||
won't be captured. | ||||
""" | ||||
Gregory Szorc
|
r37312 | def client(): | ||
Gregory Szorc
|
r36389 | """Returns a string representation of this client (as bytes).""" | ||
Gregory Szorc
|
r36631 | |||
Gregory Szorc
|
r37312 | def addcapabilities(repo, caps): | ||
Gregory Szorc
|
r36631 | """Adds advertised capabilities specific to this protocol. | ||
Receives the list of capabilities collected so far. | ||||
Returns a list of capabilities. The passed in argument can be returned. | ||||
""" | ||||
Gregory Szorc
|
r36819 | |||
Gregory Szorc
|
r37312 | def checkperm(perm): | ||
Gregory Szorc
|
r36819 | """Validate that the client has permissions to perform a request. | ||
The argument is the permission required to proceed. If the client | ||||
doesn't have that permission, the exception should raise or abort | ||||
in a protocol specific manner. | ||||
""" | ||||