2018-07-22 18:54:18 +08:00
|
|
|
// https://tools.ietf.org/html/rfc6455#section-5.2
|
|
|
|
//
|
|
|
|
// Frame Format
|
|
|
|
// 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 ... |
|
|
|
|
// +---------------------------------------------------------------+
|
2018-07-21 22:56:37 +08:00
|
|
|
|
|
|
|
package ws
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"io"
|
|
|
|
"math/rand"
|
2020-10-20 20:28:35 +08:00
|
|
|
|
|
|
|
"github.com/nadoo/glider/pool"
|
2018-07-21 22:56:37 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2020-10-21 20:00:43 +08:00
|
|
|
hdrSize = 2 + 8 // Fixed header + length
|
2018-07-23 01:41:18 +08:00
|
|
|
|
2020-10-19 20:45:57 +08:00
|
|
|
// byte 0
|
2018-07-23 01:41:18 +08:00
|
|
|
finalBit byte = 1 << 7
|
|
|
|
opCodeBinary byte = 2
|
2020-10-19 20:45:57 +08:00
|
|
|
|
|
|
|
// byte 1
|
|
|
|
maskBit byte = 1 << 7
|
2018-07-21 22:56:37 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type frameWriter struct {
|
|
|
|
io.Writer
|
2020-10-21 20:00:43 +08:00
|
|
|
header [hdrSize]byte
|
|
|
|
server bool
|
|
|
|
maskKey [4]byte
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2019-09-07 17:17:38 +08:00
|
|
|
// FrameWriter returns a frame writer.
|
2020-10-20 20:28:35 +08:00
|
|
|
func FrameWriter(w io.Writer, server bool) io.Writer {
|
2018-07-21 22:56:37 +08:00
|
|
|
n := rand.Uint32()
|
|
|
|
return &frameWriter{
|
|
|
|
Writer: w,
|
2020-10-20 20:28:35 +08:00
|
|
|
server: server,
|
2020-10-19 20:45:57 +08:00
|
|
|
maskKey: [4]byte{byte(n), byte(n >> 8), byte(n >> 16), byte(n >> 24)},
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *frameWriter) Write(b []byte) (int, error) {
|
2020-10-20 20:28:35 +08:00
|
|
|
hdr := w.header
|
|
|
|
hdr[0], hdr[1] = opCodeBinary|finalBit, 0
|
|
|
|
if !w.server {
|
|
|
|
hdr[1] = maskBit
|
|
|
|
}
|
2018-07-22 18:21:27 +08:00
|
|
|
|
2020-10-21 20:00:43 +08:00
|
|
|
nPayload, nLenField := len(b), 0
|
2020-10-20 20:28:35 +08:00
|
|
|
switch {
|
|
|
|
case nPayload <= 125:
|
|
|
|
hdr[1] |= byte(nPayload)
|
|
|
|
case nPayload < 65536:
|
|
|
|
hdr[1] |= 126
|
2020-10-21 20:00:43 +08:00
|
|
|
nLenField = 2
|
|
|
|
binary.BigEndian.PutUint16(hdr[2:2+nLenField], uint16(nPayload))
|
2020-10-20 20:28:35 +08:00
|
|
|
default:
|
|
|
|
hdr[1] |= 127
|
2020-10-21 20:00:43 +08:00
|
|
|
nLenField = 8
|
|
|
|
binary.BigEndian.PutUint64(hdr[2:2+nLenField], uint64(nPayload))
|
2020-10-20 20:28:35 +08:00
|
|
|
}
|
2020-10-19 20:45:57 +08:00
|
|
|
|
2020-10-20 20:28:35 +08:00
|
|
|
// header and length
|
2020-10-21 20:00:43 +08:00
|
|
|
_, err := w.Writer.Write(hdr[:2+nLenField])
|
2020-10-20 20:28:35 +08:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-10-19 20:45:57 +08:00
|
|
|
|
2020-10-20 20:28:35 +08:00
|
|
|
if w.server {
|
|
|
|
return w.Writer.Write(b)
|
|
|
|
}
|
2018-07-21 22:56:37 +08:00
|
|
|
|
2020-10-20 20:28:35 +08:00
|
|
|
buf := pool.GetBuffer(nPayload)
|
|
|
|
pool.PutBuffer(buf)
|
2018-07-21 22:56:37 +08:00
|
|
|
|
2020-10-21 20:00:43 +08:00
|
|
|
// mask
|
2020-10-20 20:28:35 +08:00
|
|
|
_, err = w.Writer.Write(w.maskKey[:])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2018-07-21 22:56:37 +08:00
|
|
|
|
2020-10-21 20:00:43 +08:00
|
|
|
// payload with mask
|
2020-10-20 20:28:35 +08:00
|
|
|
for i := 0; i < nPayload; i++ {
|
|
|
|
buf[i] = b[i] ^ w.maskKey[i%4]
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2020-10-20 20:28:35 +08:00
|
|
|
return w.Writer.Write(buf)
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
type frameReader struct {
|
|
|
|
io.Reader
|
2020-10-19 20:45:57 +08:00
|
|
|
buf [8]byte
|
|
|
|
left int64
|
|
|
|
server bool
|
|
|
|
maskKey [4]byte
|
|
|
|
maskOffset int
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2019-09-07 17:17:38 +08:00
|
|
|
// FrameReader returns a chunked reader.
|
2020-10-20 20:28:35 +08:00
|
|
|
func FrameReader(r io.Reader, server bool) io.Reader {
|
|
|
|
return &frameReader{Reader: r, server: server}
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *frameReader) Read(b []byte) (int, error) {
|
2020-08-25 22:14:08 +08:00
|
|
|
if r.left == 0 {
|
2018-07-23 01:41:18 +08:00
|
|
|
// get msg header
|
|
|
|
_, err := io.ReadFull(r.Reader, r.buf[:2])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2018-07-21 22:56:37 +08:00
|
|
|
|
2020-08-25 22:14:08 +08:00
|
|
|
r.left = int64(r.buf[1] & 0x7f)
|
|
|
|
switch r.left {
|
2018-07-23 01:41:18 +08:00
|
|
|
case 126:
|
2018-07-24 00:45:41 +08:00
|
|
|
_, err := io.ReadFull(r.Reader, r.buf[:2])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-08-25 22:14:08 +08:00
|
|
|
r.left = int64(binary.BigEndian.Uint16(r.buf[:2]))
|
2018-07-23 01:41:18 +08:00
|
|
|
case 127:
|
2018-07-24 00:45:41 +08:00
|
|
|
_, err := io.ReadFull(r.Reader, r.buf[:8])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2020-08-25 22:14:08 +08:00
|
|
|
r.left = int64(binary.BigEndian.Uint64(r.buf[:8]))
|
2018-07-23 01:41:18 +08:00
|
|
|
}
|
2020-10-19 20:45:57 +08:00
|
|
|
|
|
|
|
if r.server {
|
|
|
|
_, err := io.ReadFull(r.Reader, r.maskKey[:])
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
r.maskOffset = 0
|
|
|
|
}
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2018-07-24 00:45:41 +08:00
|
|
|
readLen := int64(len(b))
|
2020-08-25 22:14:08 +08:00
|
|
|
if readLen > r.left {
|
|
|
|
readLen = r.left
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2020-10-20 20:28:35 +08:00
|
|
|
m, err := io.ReadFull(r.Reader, b[:readLen])
|
2018-07-21 22:56:37 +08:00
|
|
|
if err != nil {
|
2020-10-20 20:28:35 +08:00
|
|
|
return m, err
|
2018-07-21 22:56:37 +08:00
|
|
|
}
|
|
|
|
|
2020-10-19 20:45:57 +08:00
|
|
|
if r.server {
|
|
|
|
for i := range b[:m] {
|
|
|
|
b[i] = b[i] ^ r.maskKey[(i+r.maskOffset)%4]
|
|
|
|
}
|
|
|
|
r.maskOffset = (m + r.maskOffset) % 4
|
|
|
|
}
|
|
|
|
|
2020-08-25 22:14:08 +08:00
|
|
|
r.left -= int64(m)
|
2018-07-21 22:56:37 +08:00
|
|
|
return m, err
|
|
|
|
}
|