sing-box/transport/v2raywebsocket/writer.go

88 lines
1.7 KiB
Go
Raw Normal View History

2022-09-23 09:21:24 +00:00
package v2raywebsocket
import (
"encoding/binary"
"math/rand"
"github.com/sagernet/sing/common/buf"
2022-09-30 03:27:18 +00:00
"github.com/sagernet/sing/common/bufio"
N "github.com/sagernet/sing/common/network"
2022-09-23 09:21:24 +00:00
"github.com/sagernet/websocket"
)
type Writer struct {
*websocket.Conn
2022-09-30 03:27:18 +00:00
writer N.ExtendedWriter
2022-09-23 09:21:24 +00:00
isServer bool
}
2022-09-30 03:27:18 +00:00
func NewWriter(conn *websocket.Conn, isServer bool) *Writer {
return &Writer{
conn,
bufio.NewExtendedWriter(conn.NetConn()),
isServer,
}
}
2022-09-23 09:21:24 +00:00
func (w *Writer) Write(p []byte) (n int, err error) {
err = w.Conn.WriteMessage(websocket.BinaryMessage, p)
if err != nil {
return
}
return len(p), nil
}
func (w *Writer) WriteBuffer(buffer *buf.Buffer) error {
var payloadBitLength int
dataLen := buffer.Len()
data := buffer.Bytes()
if dataLen < 126 {
payloadBitLength = 1
} else if dataLen < 65536 {
payloadBitLength = 3
} else {
payloadBitLength = 9
}
var headerLen int
headerLen += 1 // FIN / RSV / OPCODE
headerLen += payloadBitLength
if !w.isServer {
headerLen += 4 // MASK KEY
}
header := buffer.ExtendHeader(headerLen)
header[0] = websocket.BinaryMessage | 1<<7
if w.isServer {
header[1] = 0
} else {
header[1] = 1 << 7
}
if dataLen < 126 {
header[1] |= byte(dataLen)
} else if dataLen < 65536 {
header[1] |= 126
binary.BigEndian.PutUint16(header[2:], uint16(dataLen))
} else {
header[1] |= 127
binary.BigEndian.PutUint64(header[2:], uint64(dataLen))
}
if !w.isServer {
maskKey := rand.Uint32()
binary.BigEndian.PutUint32(header[1+payloadBitLength:], maskKey)
maskBytes(*(*[4]byte)(header[1+payloadBitLength:]), 0, data)
}
2022-09-30 03:27:18 +00:00
return w.writer.WriteBuffer(buffer)
}
func (w *Writer) Upstream() any {
return w.Conn.NetConn()
}
func (w *Writer) FrontHeadroom() int {
2022-10-07 12:18:50 +00:00
return 14
2022-09-23 09:21:24 +00:00
}