well-goknown/vendor/github.com/gobwas/ws/wsflate/helper.go

195 lines
5.6 KiB
Go

package wsflate
import (
"bytes"
"compress/flate"
"fmt"
"io"
"github.com/gobwas/ws"
)
// DefaultHelper is a default helper instance holding standard library's
// `compress/flate` compressor and decompressor under the hood.
//
// Note that use of DefaultHelper methods assumes that DefaultParameters were
// used for extension negotiation during WebSocket handshake.
var DefaultHelper = Helper{
Compressor: func(w io.Writer) Compressor {
// No error can be returned here as NewWriter() doc says.
f, _ := flate.NewWriter(w, 9)
return f
},
Decompressor: func(r io.Reader) Decompressor {
return flate.NewReader(r)
},
}
// DefaultParameters holds deflate extension parameters which are assumed by
// DefaultHelper to be used during WebSocket handshake.
var DefaultParameters = Parameters{
ServerNoContextTakeover: true,
ClientNoContextTakeover: true,
}
// CompressFrame is a shortcut for DefaultHelper.CompressFrame().
//
// Note that use of DefaultHelper methods assumes that DefaultParameters were
// used for extension negotiation during WebSocket handshake.
func CompressFrame(f ws.Frame) (ws.Frame, error) {
return DefaultHelper.CompressFrame(f)
}
// CompressFrameBuffer is a shortcut for DefaultHelper.CompressFrameBuffer().
//
// Note that use of DefaultHelper methods assumes that DefaultParameters were
// used for extension negotiation during WebSocket handshake.
func CompressFrameBuffer(buf Buffer, f ws.Frame) (ws.Frame, error) {
return DefaultHelper.CompressFrameBuffer(buf, f)
}
// DecompressFrame is a shortcut for DefaultHelper.DecompressFrame().
//
// Note that use of DefaultHelper methods assumes that DefaultParameters were
// used for extension negotiation during WebSocket handshake.
func DecompressFrame(f ws.Frame) (ws.Frame, error) {
return DefaultHelper.DecompressFrame(f)
}
// DecompressFrameBuffer is a shortcut for
// DefaultHelper.DecompressFrameBuffer().
//
// Note that use of DefaultHelper methods assumes that DefaultParameters were
// used for extension negotiation during WebSocket handshake.
func DecompressFrameBuffer(buf Buffer, f ws.Frame) (ws.Frame, error) {
return DefaultHelper.DecompressFrameBuffer(buf, f)
}
// Helper is a helper struct that holds common code for compression and
// decompression bytes or WebSocket frames.
//
// Its purpose is to reduce boilerplate code in WebSocket applications.
type Helper struct {
Compressor func(w io.Writer) Compressor
Decompressor func(r io.Reader) Decompressor
}
// Buffer is an interface representing some bytes buffering object.
type Buffer interface {
io.Writer
Bytes() []byte
}
// CompressFrame returns compressed version of a frame.
// Note that it does memory allocations internally. To control those
// allocations consider using CompressFrameBuffer().
func (h *Helper) CompressFrame(in ws.Frame) (f ws.Frame, err error) {
var buf bytes.Buffer
return h.CompressFrameBuffer(&buf, in)
}
// DecompressFrame returns decompressed version of a frame.
// Note that it does memory allocations internally. To control those
// allocations consider using DecompressFrameBuffer().
func (h *Helper) DecompressFrame(in ws.Frame) (f ws.Frame, err error) {
var buf bytes.Buffer
return h.DecompressFrameBuffer(&buf, in)
}
// CompressFrameBuffer compresses a frame using given buffer.
// Returned frame's payload holds bytes returned by buf.Bytes().
func (h *Helper) CompressFrameBuffer(buf Buffer, f ws.Frame) (ws.Frame, error) {
if !f.Header.Fin {
return f, fmt.Errorf("wsflate: fragmented messages are not allowed")
}
if err := h.CompressTo(buf, f.Payload); err != nil {
return f, err
}
var err error
f.Payload = buf.Bytes()
f.Header.Length = int64(len(f.Payload))
f.Header, err = SetBit(f.Header)
if err != nil {
return f, err
}
return f, nil
}
// DecompressFrameBuffer decompresses a frame using given buffer.
// Returned frame's payload holds bytes returned by buf.Bytes().
func (h *Helper) DecompressFrameBuffer(buf Buffer, f ws.Frame) (ws.Frame, error) {
if !f.Header.Fin {
return f, fmt.Errorf(
"wsflate: fragmented messages are not supported by helper",
)
}
var (
compressed bool
err error
)
f.Header, compressed, err = UnsetBit(f.Header)
if err != nil {
return f, err
}
if !compressed {
return f, nil
}
if err := h.DecompressTo(buf, f.Payload); err != nil {
return f, err
}
f.Payload = buf.Bytes()
f.Header.Length = int64(len(f.Payload))
return f, nil
}
// Compress compresses given bytes.
// Note that it does memory allocations internally. To control those
// allocations consider using CompressTo().
func (h *Helper) Compress(p []byte) ([]byte, error) {
var buf bytes.Buffer
if err := h.CompressTo(&buf, p); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
// Decompress decompresses given bytes.
// Note that it does memory allocations internally. To control those
// allocations consider using DecompressTo().
func (h *Helper) Decompress(p []byte) ([]byte, error) {
var buf bytes.Buffer
if err := h.DecompressTo(&buf, p); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
// CompressTo compresses bytes into given buffer.
func (h *Helper) CompressTo(w io.Writer, p []byte) (err error) {
c := NewWriter(w, h.Compressor)
if _, err = c.Write(p); err != nil {
return err
}
if err := c.Flush(); err != nil {
return err
}
if err := c.Close(); err != nil {
return err
}
return nil
}
// DecompressTo decompresses bytes into given buffer.
// Returned bytes are bytes returned by buf.Bytes().
func (h *Helper) DecompressTo(w io.Writer, p []byte) (err error) {
fr := NewReader(bytes.NewReader(p), h.Decompressor)
if _, err = io.Copy(w, fr); err != nil {
return err
}
if err := fr.Close(); err != nil {
return err
}
return nil
}