All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.tio.websocket.common.WsClientEncoder Maven / Gradle / Ivy

package org.tio.websocket.common;

import java.nio.ByteBuffer;
import java.util.Random;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.tio.core.ChannelContext;
import org.tio.core.GroupContext;
import org.tio.core.utils.ByteBufferUtils;

public class WsClientEncoder {
  private static Logger log = LoggerFactory.getLogger(WsClientEncoder.class);

  private static final Random reuseableRandom = new Random();

  /*
      0                   1                   2                   3
      0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
     +-+-+-+-+-------+-+-------------+-------------------------------+
     |F|R|R|R| opcode|M| Payload len |    Extended payload length    |
     |I|S|S|S|  (4)  |A|     (7)     |             (16/64)           |
     |N|V|V|V|       |S|             |   (if payload len==126/127)   |
     | |1|2|3|       |K|             |                               |
     +-+-+-+-+-------+-+-------------+ - - - - - - - - - - - - - - - +
     |     Extended payload length continued, if payload len == 127  |
     + - - - - - - - - - - - - - - - +-------------------------------+
     |                               |Masking-key, if MASK set to 1  |
     +-------------------------------+-------------------------------+
     | Masking-key (continued)       |          Payload Data         |
     +-------------------------------- - - - - - - - - - - - - - - - +
     :                     Payload Data continued ...                :
     + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
     |                     Payload Data continued ...                |
     +---------------------------------------------------------------+
  */
  public static ByteBuffer encode(
      WsPacket packet, GroupContext groupContext, ChannelContext channelContext) {
    byte[] wsBody = packet.getBody(); // 就是ws的body,不包括ws的头
    byte[][] wsBodies = packet.getBodys();
    int wsBodyLength = 0;
    if (wsBody != null) {
      wsBodyLength += wsBody.length;
    } else if (wsBodies != null) {
      for (byte[] bs : wsBodies) {
        wsBodyLength += bs.length;
      }
    }

    byte opcode = packet.getWsOpcode().getCode();
    byte b0 = (byte) (packet.isWsEof() ? -128 : 0);
    b0 |= opcode;

    byte maskedByte = (byte) -128;

    ByteBuffer buf;
    if (wsBodyLength < 126) {
      buf = ByteBuffer.allocate(2 + wsBodyLength + 4);
      buf.put(b0);
      buf.put((byte) (wsBodyLength | maskedByte));
    } else if (wsBodyLength < (1 << 16) - 1) {
      buf = ByteBuffer.allocate(4 + wsBodyLength + 4);
      buf.put(b0);
      buf.put((byte) (126 | maskedByte));
      ByteBufferUtils.writeUB2WithBigEdian(buf, wsBodyLength);
    } else {
      buf = ByteBuffer.allocate(10 + wsBodyLength + 4);
      buf.put(b0);
      buf.put((byte) (127 | maskedByte));

      buf.position(buf.position() + 4);

      ByteBufferUtils.writeUB4WithBigEdian(buf, wsBodyLength);
    }

    ByteBuffer maskkey = ByteBuffer.allocate(4);
    maskkey.putInt(reuseableRandom.nextInt());
    buf.put(maskkey.array());

    if (wsBody != null)
      for (int i = 0; i < wsBody.length; i++) {
        wsBody[i] = ((byte) (wsBody[i] ^ maskkey.get(i % 4)));
      }

    if (wsBody != null && wsBody.length > 0) {
      buf.put(wsBody);
    } else if (wsBodies != null) {
      for (byte[] bs : wsBodies) {
        buf.put(bs);
      }
    }

    return buf;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy