How to implement a custom protocol

    MetaProtocol Proxy already has most of the features needed for a Layer-7 proxy building in, including load balancing, circuit breaker, RDS dynamic routing, local/global reate limiting, header mutation, metrics and tracing reporting, etc., more features are under the way. Therefore, creating a layer-7 protocol based on MetaProtocol for a custom protocol is very simple. We only need to implement the codec interface. Normally, it’s only a few hundred lines of code.

    After implementing the data proxy based on MetaProtocol, without any control-plane coding, we can manage the services using those custom protocols in the Isito service mesh via Aeraki, providing service governance capabilities for those services, such as traffic splitting, canary releasing, traffic mirroring, etc. This is because Aeraki recognizes any Layer-7 protocol based on MetaProtocol and provides user-friendly traffic rules at the control plane. Aeraki translates these traffic rules into MetaProtocol configurations and then distribute them to the data plane.

    Another benefit of using MetaProtocol to develop custom protocols in a service mesh is that the solution is completely non-intrusive to the upstream open source projects such as Istio, Envoy, and MetaProtocol Proxy itself. It allows rapid iterations to follow the upstream projects and take advantage of the new capabilities provided by new versions of the upstream projects without maintaining your own forked branches.

    Aeraki provides a scaffold project awesomerpc to demonstrate how to implement a custom protocol with MetaProtocol proxy. You can fork this repo and use it as the start point to build a layer-7 proxy for your proprietary protocol.

    The most important files are and awesomerpc_codec.cc in the src/application_protocols/awesomerpc/ directory. These two files define the interface and implementation code for the codec of the application protocol.

    1. /**
    2. * Codec for Awesomerpc protocol.
    3. */
    4. class AwesomerpcCodec : public MetaProtocolProxy::Codec,
    5. public Logger::Loggable<Logger::Id::misc> {
    6. public:
    7. AwesomerpcCodec() {};
    8. ~AwesomerpcCodec() override = default;
    9. //For protocol decoding, the buffer needs to be parsed and populated with Metadata, which will be used for MetaProtocol Proxy filters, such as flow restriction and matching conditions for routes.
    10. MetaProtocolProxy::DecodeStatus decode(Buffer::Instance& buffer,
    11. MetaProtocolProxy::Metadata& metadata) override;
    12. //Protocol encoding, the request or response packet can be modified according to Mutation, such as adding, deleting or modifying the header, which needs to be written back to the buffer after modification
    13. void encode(const MetaProtocolProxy::Metadata& metadata,
    14. //the framework uses err code to return error messages to the client, such as route not found or connection creation failure, etc., the encoded data needs to be written to the buffer
    15. void onError(const MetaProtocolProxy::Metadata& metadata, const MetaProtocolProxy::Error& error,
    16. Buffer::Instance& buffer) override;
    17. ...

    You can also refer to the and thrift implementation when writing the codec.

    Configure the metaProtocol, envoy and Istio-Proxy dependencies in the WORKSPACE file in the root directory.

    You need to configure the git commit of metaProtocol in WORKSPACE, and the dependencies of envoy and Istio-Proxy refer to the configuration in WORKSPACE in the commit of metaProtocol. See for version dependencies.

    It is recommended to use Ubuntu 18.04 as the build environment.

    Install the required software for compiling:

    1. sudo wget -O /usr/local/bin/bazel https://github.com/bazelbuild/bazelisk/releases/latest/download/bazelisk-linux-$([ $(uname -m) = "aarch64" ] && echo "arm64" || echo "amd64")
    2. sudo chmod +x /usr/local/bin/bazel
    3. sudo apt-get install \
    4. autoconf \
    5. automake \
    6. cmake \
    7. curl \
    8. libtool \
    9. make \
    10. ninja-build \
    11. patch \
    12. python3-pip \
    13. unzip \
    14. virtualenv
    15. sudo apt update
    16. sudo apt-get install llvm-10 lldb-10 llvm-10-dev libllvm10 llvm-10-runtime clang-10 clang++-10 lld-10 gcc-10 g++-10

    To identify custom protocols in Istio, you need to create an Aeraki’s ApplicationProtocol CRD resource.

    1. apiVersion: metaprotocol.aeraki.io/v1alpha1
    2. kind: ApplicationProtocol
    3. metadata:
    4. name: my-protocol
    5. namespace: istio-system
    6. spec:
    7. protocol: my-protocol
    8. codec: aeraki.meta_protocol.codec.my_protocol

    Aeraki identifies MetaProtocol-based application protocols by the service prefix, and the port name of the service must follow this naming convention: tcp-metaprotocol-{application protocol}-xxx. Service definitions can be K8s service or Istio ServiceEntry.

    The following ServiceEntry defines a dubbo service:

    The following Service defines a Thrift service:

    1. apiVersion: v1
    2. kind: Service
    3. metadata:
    4. name: thrift-sample-server
    5. spec:
    6. selector:
    7. app: thrift-sample-server
    8. ports:
    9. - name: tcp-metaprotocol-thrift-hello-server
    10. protocol: TCP
    11. port: 9090