症结在于负载 > 125 字节。
格式非常简单,假设您在 JavaScript 中发送十个 a:
ws.send("a".repeat(10))
然后服务器会收到:
bytes[16]=818a8258a610e339c771e339c771e339
- 字节0:0x81只是一个消息接收的指示符
- 字节1:0x8a是长度,减去0x80,0x0A == 10
- 字节 2, 3, 4, 5:用于解密有效负载的 4 字节异或密钥
- 其余:有效载荷
但现在假设您在 JavaScript 中发送 126 个 a:
ws.send("a".repeat(126))
然后服务器会收到:
bytes[134]=81fe007ee415f1e5857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574908485749084857490848574
如果有效负载的长度 > 125,则字节 1 的值为 0xfe,格式将更改为:
- 字节0:0x81只是一个消息接收的指示符
- 字节 1:将为 0xfe
- 字节 2、3:作为 uint16 数字的有效负载的长度
- 字节 4, 5, 6, 7:用于解密有效负载的 4 字节异或密钥
- 其余:有效载荷
C# 中的示例代码:
List<byte[]> decodeWebsocketFrame(Byte[] bytes)
{
List<Byte[]> ret = new List<Byte[]>();
int offset = 0;
while (offset + 6 < bytes.Length)
{
// format: 0==ascii/binary 1=length-0x80, byte 2,3,4,5=key, 6+len=message, repeat with offset for next...
int len = bytes[offset + 1] - 0x80;
if (len <= 125)
{
//String data = Encoding.UTF8.GetString(bytes);
//Debug.Log("len=" + len + "bytes[" + bytes.Length + "]=" + ByteArrayToString(bytes) + " data[" + data.Length + "]=" + data);
Debug.Log("len=" + len + " offset=" + offset);
Byte[] key = new Byte[] { bytes[offset + 2], bytes[offset + 3], bytes[offset + 4], bytes[offset + 5] };
Byte[] decoded = new Byte[len];
for (int i = 0; i < len; i++)
{
int realPos = offset + 6 + i;
decoded[i] = (Byte)(bytes[realPos] ^ key[i % 4]);
}
offset += 6 + len;
ret.Add(decoded);
} else
{
int a = bytes[offset + 2];
int b = bytes[offset + 3];
len = (a << 8) + b;
//Debug.Log("Length of ws: " + len);
Byte[] key = new Byte[] { bytes[offset + 4], bytes[offset + 5], bytes[offset + 6], bytes[offset + 7] };
Byte[] decoded = new Byte[len];
for (int i = 0; i < len; i++)
{
int realPos = offset + 8 + i;
decoded[i] = (Byte)(bytes[realPos] ^ key[i % 4]);
}
offset += 8 + len;
ret.Add(decoded);
}
}
return ret;
}