diff options
| author | Maximilian Hils <git@maximilianhils.com> | 2015-09-03 18:55:38 +0200 | 
|---|---|---|
| committer | Maximilian Hils <git@maximilianhils.com> | 2015-09-03 18:55:38 +0200 | 
| commit | 14457f29b3d89e234d0791c4980e5cf9514185dd (patch) | |
| tree | 9a12632fea816eb5a7f701d3e92670dc4009ccfe /libmproxy/protocol/base.py | |
| parent | 99126f62ed947847eba4cfa687cb0b0f012092bb (diff) | |
| download | mitmproxy-14457f29b3d89e234d0791c4980e5cf9514185dd.tar.gz mitmproxy-14457f29b3d89e234d0791c4980e5cf9514185dd.tar.bz2 mitmproxy-14457f29b3d89e234d0791c4980e5cf9514185dd.zip | |
docs++
Diffstat (limited to 'libmproxy/protocol/base.py')
| -rw-r--r-- | libmproxy/protocol/base.py | 116 | 
1 files changed, 71 insertions, 45 deletions
| diff --git a/libmproxy/protocol/base.py b/libmproxy/protocol/base.py index f27cb04b..9d8c8bfe 100644 --- a/libmproxy/protocol/base.py +++ b/libmproxy/protocol/base.py @@ -1,37 +1,3 @@ -""" -mitmproxy protocol architecture - -In mitmproxy, protocols are implemented as a set of layers, which are composed on top each other. -For example, the following scenarios depict possible settings (lowest layer first): - -Transparent HTTP proxy, no SSL: -    TransparentProxy -    Http1Layer -    HttpLayer - -Regular proxy, CONNECT request with WebSockets over SSL: -    HttpProxy -    Http1Layer -    HttpLayer -    SslLayer -    WebsocketLayer (or TcpLayer) - -Automated protocol detection by peeking into the buffer: -    TransparentProxy -    TLSLayer -    Http2Layer -    HttpLayer - -Communication between layers is done as follows: -    - lower layers provide context information to higher layers -    - higher layers can call functions provided by lower layers, -      which are propagated until they reach a suitable layer. - -Further goals: -  - Connections should always be peekable to make automatic protocol detection work. -  - Upstream connections should be established as late as possible; -    inline scripts shall have a chance to handle everything locally. -"""  from __future__ import (absolute_import, print_function, division)  from netlib import tcp  from ..models import ServerConnection @@ -43,8 +9,8 @@ class _LayerCodeCompletion(object):      Dummy class that provides type hinting in PyCharm, which simplifies development a lot.      """ -    def __init__(self, *args, **kwargs):  # pragma: nocover -        super(_LayerCodeCompletion, self).__init__(*args, **kwargs) +    def __init__(self, **mixin_args):  # pragma: nocover +        super(_LayerCodeCompletion, self).__init__(**mixin_args)          if True:              return          self.config = None @@ -55,34 +21,64 @@ class _LayerCodeCompletion(object):          """@type: libmproxy.models.ServerConnection"""          self.channel = None          """@type: libmproxy.controller.Channel""" +        self.ctx = None +        """@type: libmproxy.protocol.Layer"""  class Layer(_LayerCodeCompletion): -    def __init__(self, ctx, *args, **kwargs): +    """ +    Base class for all layers. All other protocol layers should inherit from this class. +    """ + +    def __init__(self, ctx, **mixin_args):          """ +        Each layer usually passes itself to its child layers as a context. Properties of the +        context are transparently mapped to the layer, so that the following works: + +        .. code-block:: python + +            root_layer = Layer(None) +            root_layer.client_conn = 42 +            sub_layer = Layer(root_layer) +            print(sub_layer.client_conn) # 42 + +        The root layer is passed a :py:class:`libmproxy.proxy.RootContext` object, +        which provides access to :py:attr:`.client_conn <libmproxy.proxy.RootContext.client_conn>`, +        :py:attr:`.next_layer <libmproxy.proxy.RootContext.next_layer>` and other basic attributes. +          Args: -            ctx: The (read-only) higher layer. +            ctx: The (read-only) parent layer / context.          """          self.ctx = ctx -        """@type: libmproxy.protocol.Layer""" -        super(Layer, self).__init__(*args, **kwargs) +        """ +        The parent layer. -    def __call__(self): +        :type: :py:class:`Layer`          """ -        Logic of the layer. +        super(Layer, self).__init__(**mixin_args) + +    def __call__(self): +        """Logic of the layer. + +        Returns: +            Once the protocol has finished without exceptions. +          Raises: -            ProtocolException in case of protocol exceptions. +            ~libmproxy.exceptions.ProtocolException: if an exception occurs. No other exceptions must be raised.          """          raise NotImplementedError()      def __getattr__(self, name):          """ -        Attributes not present on the current layer may exist on a higher layer. +        Attributes not present on the current layer are looked up on the context.          """          return getattr(self.ctx, name)      @property      def layers(self): +        """ +        List of all layers, including the current layer (``[self, self.ctx, self.ctx.ctx, ...]``) +        """          return [self] + self.ctx.layers      def __repr__(self): @@ -92,6 +88,20 @@ class Layer(_LayerCodeCompletion):  class ServerConnectionMixin(object):      """      Mixin that provides a layer with the capabilities to manage a server connection. +    The server address can be passed in the constructor or set by calling :py:meth:`set_server`. +    Subclasses are responsible for calling :py:meth:`disconnect` before returning. + +    Recommended Usage: + +    .. code-block:: python + +        class MyLayer(Layer, ServerConnectionMixin): +            def __call__(self): +                try: +                    # Do something. +                finally: +                    if self.server_conn: +                        self.disconnect()      """      def __init__(self, server_address=None): @@ -117,6 +127,14 @@ class ServerConnectionMixin(object):                  )      def set_server(self, address, server_tls=None, sni=None): +        """ +        Sets a new server address. If there is an existing connection, it will be closed. + +        Raises: +            ~libmproxy.exceptions.ProtocolException: +                if ``server_tls`` is ``True``, but there was no TLS layer on the +                protocol stack which could have processed this. +        """          if self.server_conn:              self.disconnect()          self.log("Set new server address: " + repr(address), "debug") @@ -130,6 +148,7 @@ class ServerConnectionMixin(object):      def disconnect(self):          """          Deletes (and closes) an existing server connection. +        Must not be called if there is no existing connection.          """          self.log("serverdisconnect", "debug", [repr(self.server_conn.address)])          address = self.server_conn.address @@ -139,6 +158,13 @@ class ServerConnectionMixin(object):          self.server_conn = ServerConnection(address)      def connect(self): +        """ +        Establishes a server connection. +        Must not be called if there is an existing connection. + +        Raises: +            ~libmproxy.exceptions.ProtocolException: if the connection could not be established. +        """          if not self.server_conn.address:              raise ProtocolException("Cannot connect to server, no server address given.")          self.log("serverconnect", "debug", [repr(self.server_conn.address)]) @@ -152,5 +178,5 @@ class ServerConnectionMixin(object):  class Kill(Exception):      """ -    Kill a connection. +    Signal that both client and server connection(s) should be killed immediately.      """ | 
