Netty - Sticky Packets and Half Packets (Part 2)

Netty - Sticky Packets and Half Packets (Part 2)

Continue from the previous article "Introduction and Solution of TCP Sticky Packet and Half Packet (Part 1)"

The previous article introduced sticky packets and half packets and their general solutions. Today, let’s focus on how Netty implements the encapsulation framing solution.

Decoding core process

Previously, we introduced three decoders: FixedLengthFrameDecoder, DelimiterBasedFrameDecoder, and LengthFieldBasedFrameDecoder. They all inherit from ByteToMessageDecoder, which inherits from ChannelInboundHandlerAdapter, and its core method is channelRead. Therefore, let's take a look at the channelRead method of ByteToMessageDecoder:

  1. @Override
  2. public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
  3. if (msg instanceof ByteBuf) {
  4. CodecOutputList out = CodecOutputList .newInstance();
  5. try {
  6. //Convert the incoming message into data
  7. ByteBuf data = (ByteBuf) msg;
  8. // The ultimate goal is to put all the data into cumulation
  9. first = cumulation == null;
  10. // The first data is put directly into
  11. if (first) {
  12. cumulation = data ;
  13. } else {
  14. // If it is not the first data, append
  15. cumulation = cumulator .cumulate(ctx.alloc(), cumulation, data);
  16. }
  17. // Decode
  18. callDecode(ctx, cumulation, out);
  19. }
  20. // The following code is omitted because it does not belong to the decoding process
  21. }

Let's take a look at the callDecode method:

  1. protected void callDecode(ChannelHandlerContext ctx, ByteBuf in, List < Object > out) {
  2. try {
  3. while (in.isReadable()) {
  4. int out outSize = out.size();
  5. if (outSize > 0) {
  6. // The following code is omitted because in the initial state, outSize can only be 0 and it is impossible to enter here
  7. }
  8. int oldInputLength = in .readableBytes();
  9. // When decoding, the handler's remove operation is not executed.
  10. // Only after decode is executed, start cleaning data.
  11. decodeRemovalReentryProtection(ctx, in, out);
  12. // Omit the following code because the following content is not the decoding process

Let's take a look at the decodeRemovalReentryProtection method:

  1. final void decodeRemovalReentryProtection(ChannelHandlerContext ctx, ByteBuf in, List < Object > out)
  2. throws Exception {
  3. // Set the current state to decoding
  4. decodeState = STATE_CALLING_CHILD_DECODE ;
  5. try {
  6. // Decode
  7. decode(ctx, in, out);
  8. finally
  9. // Execute the remove operation of hander
  10. boolean removePending = decodeState == STATE_HANDLER_REMOVED_PENDING;
  11. decodeState = STATE_INIT ;
  12. if (removePending) {
  13. handlerRemoved(ctx);
  14. }
  15. }
  16. }
  17. // Subclasses override this method, and each implementation has its own special decoding method
  18. protected abstract void decode(ChannelHandlerContext ctx, ByteBuf in, List < Object > out) throws Exception;

From the above process, we can conclude that before decoding, the data needs to be written into the cumulation, and after decoding, it needs to be removed through the handler.

Specific decoding process

I just mentioned that the decode method is implemented in the subclass. Let's look at the implementation of the three decoding methods we mentioned one by one.

1. FixedLengthFrameDecoder

The source code is:

  1. @Override
  2. protected final void decode(ChannelHandlerContext ctx, ByteBuf in, List < Object > out) throws Exception {
  3. Object decode decoded = decode(ctx, in);
  4. if (decoded != null) {
  5. out.add(decoded);
  6. }
  7. }
  8. protected Object decode(
  9. @SuppressWarnings("UnusedParameters") ChannelHandlerContext ctx, ByteBuf in) throws Exception {
  10. // Whether the collected data is less than the fixed length, if it is less than, it means that it cannot be parsed
  11. if (in.readableBytes() <   frameLength ) {
  12. return null;
  13. } else {
  14. return in.readRetainedSlice(frameLength);
  15. }
  16. }

As simple as the name of this class, it decodes with a fixed length. Therefore, when setting up the decoder, you need to pass in the frameLength in the construction method.

2. DelimiterBasedFrameDecoder

The source code is:

  1. @Override
  2. protected final void decode(ChannelHandlerContext ctx, ByteBuf in, List < Object > out) throws Exception {
  3. Object decode decoded = decode(ctx, in);
  4. if (decoded != null) {
  5. out.add(decoded);
  6. }
  7. }
  8. protected Object decode(ChannelHandlerContext ctx, ByteBuf buffer) throws Exception {
  9. // Is the current delimiter a line break delimiter (\n or \r\n)?
  10. if (lineBasedDecoder != null) {
  11. return lineBasedDecoder.decode(ctx, buffer);
  12. }
  13. // Try all delimiters and choose the delimiter which yields the shortest frame.
  14. int minFrameLength = Integer .MAX_VALUE;
  15. ByteBuf minDelim = null ;
  16. // Other delimiters are used for splitting
  17. for (ByteBuf delim: delimiters) {
  18. int frameLength = indexOf (buffer, delim);
  19. if (frameLength > = 0 && frameLength <   minFrameLength ) {
  20. minFrameLength = frameLength ;
  21. minDelim = delim ;
  22. }
  23. }
  24. // The following code is omitted

As its name suggests, the delimiter is its core. It divides delimiters into two categories, only line break delimiters (n or rn) and others. Therefore, it is important to note that you can define multiple delimiters, and it supports them all.

3. LengthFieldBasedFrameDecoder

This class is quite complex, and it is easy to get confused if you look at the methods directly, so I plan to combine the explanation of the class and look at its private variables first.

2 bytes length field at offset 1 in the middle of 4 bytes header, strip the first header field and the length field, the length field represents the length of the whole message

Let's give another twist to the previous example. The only difference from the previous example is that the length field represents the length of the whole message instead of the message body, just like the third example. We have to count the length of HDR1 and Length into lengthAdjustment. Please note that we don't need to take the length of HDR2 into account because the length field already includes the whole header length.

  1. * BEFORE DECODE (16 bytes) AFTER DECODE (13 bytes)
  2. * +------+--------+------+----------------+ +------+----------------+
  3. * | HDR1 | Length | HDR2 | Actual Content |----- > | HDR2 | Actual Content |
  4. * | 0xCA | 0x0010 | 0xFE | "HELLO, WORLD" | | 0xFE | "HELLO, WORLD" |
  5. * +------+--------+------+----------------+ +------+----------------+
  • lengthFieldOffset: This field indicates the byte from which the Length field starts. In the above example, the Length field starts from the first byte (HDR1 is the 0th byte), so the value is 0.
  • lengthFieldLength: This field represents the number of bytes occupied by the Length field. In the above example, the Length field occupies 2 bytes, so the value is 2.
  • lengthAdjustment: This field represents the distance from the end of the Length field to the actual start of the content. In the above example, because the Length field means the entire message (including HDR1, Length, HDR2, and Actual Content, and generally Length refers only to Actual Content), the distance from the end of Length to the actual start of the content (the beginning of HDR1) is equivalent to reducing 3 bytes, so it is -3.
  • initialBytesToStrip: How many bytes should be skipped from the end of the Length field when displaying. In the above example, because the actual content starts from HDR1 and the final displayed content starts from HDR2, there is a difference of 3 bytes in the middle, so the value is 3.

The decoding method of this class is relatively complicated. Students who are interested can try to analyze it by themselves.

Summarize

This article mainly explains the three ways of framing in Netty in combination with the source code in Netty. I believe you must have a different understanding.

<<:  How edge computing will benefit from 5G technology

>>:  Perhaps, it is easier to understand HTTPS this way!

Recommend

Data Cabling: How to Plan Ahead?

Data cabling is an important channel for enterpri...

Bypassing 5G and heading straight for 6G, Russia made an "incredible" decision

Russia made an incredible decision - abandoning 5...

Comparative Analysis of Kubernetes Network Plugins (Flannel, Calico, Weave)

[[269494]] This article will focus on exploring a...

How people cope with self-managed data centers

Self-managing data centers, sometimes called self...

DesiVPS: San Jose VPS starts at $18.99 per year, 1GB/25GB/1Gbps unlimited data

DesiVPS previously provided VPS hosts in Los Ange...

In addition to speed, 5G can also change these aspects of your life!

5G is the hottest buzzword at the moment, and it ...

Supply Chain Management Is Critical to SD-WAN

SD-WAN is not an all-encompassing solution; it is...