glider/proxy/ws/frame.go

211 lines
4.6 KiB
Go
Raw Normal View History

// 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 (
"bytes"
"encoding/binary"
"io"
"math/rand"
)
const (
2020-10-19 20:45:57 +08:00
defaultFrameSize = 4096
maxHeaderSize = 2 + 8 + 4 // Fixed header + length + mask
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
buf []byte
2020-10-19 20:45:57 +08:00
client 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-19 20:45:57 +08:00
func FrameWriter(w io.Writer, client bool) io.Writer {
2018-07-21 22:56:37 +08:00
n := rand.Uint32()
return &frameWriter{
Writer: w,
2020-10-19 20:45:57 +08:00
buf: make([]byte, maxHeaderSize+defaultFrameSize),
client: client,
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) {
n, err := w.ReadFrom(bytes.NewBuffer(b))
return int(n), err
}
func (w *frameWriter) ReadFrom(r io.Reader) (n int64, err error) {
for {
buf := w.buf
2020-10-19 20:45:57 +08:00
payloadBuf := buf[maxHeaderSize:]
2018-07-21 22:56:37 +08:00
nr, er := r.Read(payloadBuf)
2020-10-19 20:45:57 +08:00
2018-07-21 22:56:37 +08:00
if nr > 0 {
n += int64(nr)
2020-10-19 20:45:57 +08:00
buf[0] = opCodeBinary
buf[1] = 0
if w.client {
buf[0] |= finalBit
buf[1] = maskBit
}
2018-07-21 22:56:37 +08:00
lengthFieldLen := 0
switch {
case nr <= 125:
buf[1] |= byte(nr)
case nr < 65536:
buf[1] |= 126
lengthFieldLen = 2
binary.BigEndian.PutUint16(buf[2:2+lengthFieldLen], uint16(nr))
default:
buf[1] |= 127
lengthFieldLen = 8
binary.BigEndian.PutUint64(buf[2:2+lengthFieldLen], uint64(nr))
}
2018-07-23 01:41:18 +08:00
// header and length
_, ew := w.Writer.Write(buf[:2+lengthFieldLen])
if ew != nil {
err = ew
break
}
payloadBuf = payloadBuf[:nr]
2020-10-19 20:45:57 +08:00
if w.client {
// maskkey
_, ew = w.Writer.Write(w.maskKey[:])
if ew != nil {
err = ew
break
}
// payload mask
for i := range payloadBuf {
payloadBuf[i] = payloadBuf[i] ^ w.maskKey[i%4]
}
2018-07-21 22:56:37 +08:00
}
_, ew = w.Writer.Write(payloadBuf)
2018-07-21 22:56:37 +08:00
if ew != nil {
err = ew
break
}
}
if er != nil {
if er != io.EOF { // ignore EOF as per io.ReaderFrom contract
err = er
}
break
}
}
return n, err
}
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-19 20:45:57 +08:00
func FrameReader(r io.Reader, client bool) io.Reader {
return &frameReader{Reader: r, server: !client}
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 {
2020-10-19 20:45:57 +08:00
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
2018-07-23 01:41:18 +08:00
// final := r.buf[0]&finalBit != 0
// frameType := int(r.buf[0] & 0xf)
2020-10-19 20:45:57 +08:00
// r.mask = r.buf[1]&maskBit != 0
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
}
2018-07-24 00:45:41 +08:00
m, err := r.Reader.Read(b[:readLen])
2018-07-21 22:56:37 +08:00
if err != nil {
return 0, err
}
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
}