Documentation ¶
Overview ¶
package msutil provides utilities for working with WebSocket protocol.
Overview:
// Read masked text message from peer and check utf8 encoding. header, err := ms.ReadHeader(conn) if err != nil { // handle err } // Prepare to read payload. r := io.LimitReader(conn, header.Length) r = msutil.NewCipherReader(r, header.Mask) r = msutil.NewUTF8Reader(r) payload, err := io.ReadAll(r) if err != nil { // handle err }
You could get the same behavior using just `msutil.Reader`:
r := msutil.Reader{ Source: conn, CheckUTF8: true, } payload, err := io.ReadAll(r) if err != nil { // handle err }
Or even simplest:
payload, err := msutil.ReadClientText(conn) if err != nil { // handle err }
Package is also exports tools for buffered writing:
// Create buffered writer, that will buffer output bytes and send them as // 128-length fragments (with exception on large writes, see the doc). writer := msutil.NewWriterSize(conn, ms.StateServerSide, ms.OpText, 128) _, err := io.CopyN(writer, rand.Reader, 100) if err == nil { err = writer.Flush() } if err != nil { // handle error }
For more utils and helpers see the documentation.
Index ¶
- Variables
- func ConnectClient(ctx context.Context, conn net.Conn, session ms.ClientHandler, log ms.Logger) error
- func ConnectServer(ctx context.Context, addr string, session ms.ClientHandler, log ms.Logger) error
- func DialServer(addr string) (net.Conn, error)
- func HandleClientControlMessage(conn io.Writer, msg Message) error
- func HandleControlMessage(conn io.Writer, state ms.State, msg Message) error
- func HandleServerControlMessage(conn io.Writer, msg Message) error
- func NextReader(r io.Reader, s ms.State) (ms.Header, io.Reader, error)
- func PutWriter(w *Writer)
- func ReadClientBinary(rw io.ReadWriter) ([]byte, error)
- func ReadClientData(rw io.ReadWriter) ([]byte, ms.OpCode, error)
- func ReadClientText(rw io.ReadWriter) ([]byte, error)
- func ReadData(rw io.ReadWriter, s ms.State) ([]byte, ms.OpCode, error)
- func ReadServerBinary(rw io.ReadWriter) ([]byte, error)
- func ReadServerData(rw io.ReadWriter) ([]byte, ms.OpCode, error)
- func ReadServerText(rw io.ReadWriter) ([]byte, error)
- func WriteClientBinary(w io.Writer, p []byte) error
- func WriteClientMessage(w io.Writer, op ms.OpCode, p []byte) error
- func WriteClientText(w io.Writer, p []byte) error
- func WriteMessage(w io.Writer, s ms.State, op ms.OpCode, p []byte) error
- func WriteServerBinary(w io.Writer, p []byte) error
- func WriteServerMessage(w io.Writer, op ms.OpCode, p []byte) error
- func WriteServerText(w io.Writer, p []byte) error
- type AutoConnectClient
- type CipherReader
- type CipherWriter
- type Client
- type ClosedError
- type Connecter
- type ControlHandler
- type ControlWriter
- type DebugUpgrader
- type FrameHandlerFunc
- type Message
- type Reader
- type RecvExtension
- type RecvExtensionFunc
- type SendExtension
- type SendExtensionFunc
- type UTF8Reader
- type Writer
- func GetWriter(dest io.Writer, state ms.State, op ms.OpCode, n int) *Writer
- func NewWriter(dest io.Writer, state ms.State, op ms.OpCode) *Writer
- func NewWriterBuffer(dest io.Writer, state ms.State, op ms.OpCode, buf []byte) *Writer
- func NewWriterBufferSize(dest io.Writer, state ms.State, op ms.OpCode, n int) *Writer
- func NewWriterSize(dest io.Writer, state ms.State, op ms.OpCode, n int) *Writer
- func (w *Writer) Available() int
- func (w *Writer) Buffered() int
- func (w *Writer) DisableFlush()
- func (w *Writer) Flush() error
- func (w *Writer) FlushFragment() error
- func (w *Writer) Grow(n int)
- func (w *Writer) ReadFrom(src io.Reader) (n int64, err error)
- func (w *Writer) Reset(dest io.Writer, state ms.State, op ms.OpCode)
- func (w *Writer) ResetOp(op ms.OpCode)
- func (w *Writer) SetExtensions(xs ...SendExtension)
- func (w *Writer) Size() int
- func (w *Writer) Write(p []byte) (n int, err error)
- func (w *Writer) WriteThrough(p []byte) (n int, err error)
Constants ¶
This section is empty.
Variables ¶
var ( ErrNoURL = errors.New("frame socket is no url config") ErrClientClosed = errors.New("client connect closed") )
var ( // ErrNotEmpty is returned by Writer.WriteThrough() to indicate that buffer is // not empty and write through could not be done. That is, caller should call // Writer.FlushFragment() to make buffer empty. ErrNotEmpty = fmt.Errorf("writer not empty") // ErrControlOverflow is returned by ControlWriter.Write() to indicate that // no more data could be written to the underlying io.Writer because // MaxControlFramePayloadSize limit is reached. ErrControlOverflow = fmt.Errorf("control frame payload overflow") )
var DefaultWriteBuffer = 4096
DefaultWriteBuffer contains size of Writer's default buffer. It used by Writer constructor functions.
var ErrFrameTooLarge = errors.New("frame too large")
ErrFrameTooLarge indicates that a message of length higher than MaxFrameSize was being read.
var ErrInvalidUTF8 = fmt.Errorf("invalid utf8")
ErrInvalidUTF8 is returned by UTF8 reader on invalid utf8 sequence.
var ErrNoFrameAdvance = errors.New("no frame advance")
ErrNoFrameAdvance means that Reader's Read() method was called without preceding NextFrame() call.
var ErrNotControlFrame = errors.New("not a control frame")
ErrNotControlFrame is returned by ControlHandler to indicate that given header could not be handled.
Functions ¶
func ConnectClient ¶
func ConnectServer ¶
func HandleClientControlMessage ¶
HandleClientControlMessage handles control frame from conn and writes response when needed.
It considers that caller represents server side.
func HandleControlMessage ¶
HandleControlMessage handles message which was read by ReadMessage() functions.
That is, it is expected, that payload is already unmasked and frame header were checked by ms.CheckHeader() call.
func HandleServerControlMessage ¶
HandleServerControlMessage handles control frame from conn and writes response when needed.
It considers that caller represents client side.
func NextReader ¶
NextReader prepares next message read from r. It returns header that describes the message and io.Reader to read message's payload. It returns non-nil error when it is not possible to read message's initial frame.
Note that next NextReader() on the same r should be done after reading all bytes from previously returned io.Reader. For more performant way to discard message use Reader and its Discard() method.
Note that it will not handle any "intermediate" frames, that possibly could be received between text/binary continuation frames. That is, if peer sent text/binary frame with fin flag "false", then it could send ping frame, and eventually remaining part of text/binary frame with fin "true" – with NextReader() the ping frame will be dropped without any notice. To handle this rare, but possible situation (and if you do not know exactly which frames peer could send), you could use Reader with OnIntermediate field set.
func ReadClientBinary ¶
func ReadClientBinary(rw io.ReadWriter) ([]byte, error)
ReadClientBinary reads next binary message from rw, considering that caller represents server side. It is a shortcut for ReadData(rw, ms.StateServerSide). It discards received text messages.
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadClientData ¶
ReadClientData reads next data message from rw, considering that caller represents server side. It is a shortcut for ReadData(rw, ms.StateServerSide).
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadClientText ¶
func ReadClientText(rw io.ReadWriter) ([]byte, error)
ReadClientText reads next text message from rw, considering that caller represents server side. It is a shortcut for ReadData(rw, ms.StateServerSide). It discards received binary messages.
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadData ¶
ReadData is a helper function that reads next data (non-control) message from rw. It takes care on handling all control frames. It will write response on control frames to the write part of rw. It blocks until some data frame will be received.
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadServerBinary ¶
func ReadServerBinary(rw io.ReadWriter) ([]byte, error)
ReadServerBinary reads next binary message from rw, considering that caller represents client side. It is a shortcut for ReadData(rw, ms.StateClientSide). It discards received text messages.
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadServerData ¶
ReadServerData reads next data message from rw, considering that caller represents client side. It is a shortcut for ReadData(rw, ms.StateClientSide).
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func ReadServerText ¶
func ReadServerText(rw io.ReadWriter) ([]byte, error)
ReadServerText reads next text message from rw, considering that caller represents client side. It is a shortcut for ReadData(rw, ms.StateClientSide). It discards received binary messages.
Note this may handle and write control frames into the writer part of a given io.ReadWriter.
func WriteClientBinary ¶
WriteClientBinary is the same as WriteClientMessage with ms.OpBinary.
func WriteClientMessage ¶
WriteClientMessage writes message to w, considering that caller represents client side.
func WriteClientText ¶
WriteClientText is the same as WriteClientMessage with ms.OpText.
func WriteMessage ¶
WriteMessage is a helper function that writes message to the w. It constructs single frame with given operation code and payload. It uses given state to prepare side-dependent things, like cipher payload bytes from client to server. It will not mutate p bytes if cipher must be made.
If you want to write message in fragmented frames, use Writer instead.
func WriteServerBinary ¶
WriteServerBinary is the same as WriteServerMessage with ms.OpBinary.
func WriteServerMessage ¶
WriteServerMessage writes message to w, considering that caller represents server side.
Types ¶
type AutoConnectClient ¶
type AutoConnectClient struct { *sync.Mutex AutoReconnectErrors int // contains filtered or unexported fields }
func NewAutoConnectClient ¶
func NewAutoConnectClient(session ms.ClientHandler, addr string, log ms.Logger) *AutoConnectClient
func (*AutoConnectClient) Run ¶
func (c *AutoConnectClient) Run(ctx context.Context, cancel context.CancelFunc)
type CipherReader ¶
type CipherReader struct {
// contains filtered or unexported fields
}
CipherReader implements io.Reader that applies xor-cipher to the bytes read from source. It could help to unmask WebSocket frame payload on the fly.
func NewCipherReader ¶
func NewCipherReader(r io.Reader, mask [4]byte) *CipherReader
NewCipherReader creates xor-cipher reader from r with given mask.
type CipherWriter ¶
type CipherWriter struct {
// contains filtered or unexported fields
}
CipherWriter implements io.Writer that applies xor-cipher to the bytes written to the destination writer. It does not modify the original bytes.
func NewCipherWriter ¶
func NewCipherWriter(w io.Writer, mask [4]byte) *CipherWriter
NewCipherWriter creates xor-cipher writer to w with given mask.
type ClosedError ¶
type ClosedError struct { Code ms.StatusCode Reason string }
ClosedError returned when peer has closed the connection with appropriate code and a textual reason.
type Connecter ¶
type Connecter struct { ms.SessionsHandler // contains filtered or unexported fields }
func NewConnecter ¶
func NewConnecter(sections ms.SessionsHandler, log ms.Logger) *Connecter
type ControlHandler ¶
type ControlHandler struct { Src io.Reader Dst io.Writer State ms.State // DisableSrcCiphering disables unmasking payload data read from Src. // It is useful when msutil.Reader is used or when frame payload already // pulled and ciphered out from the connection (and introduced by // bytes.Reader, for example). DisableSrcCiphering bool }
ControlHandler contains logic of handling control frames.
The intentional way to use it is to read the next frame header from the connection, optionally check its validity via ms.CheckHeader() and if it is not a ms.OpText of ms.OpBinary (or ms.OpContinuation) – pass it to Handle() method.
That is, passed header should be checked to get rid of unexpected errors.
The Handle() method will read out all control frame payload (if any) and write necessary bytes as a rfc compatible response.
func (ControlHandler) Handle ¶
func (c ControlHandler) Handle(h ms.Header) error
Handle handles control frames regarding to the c.State and writes responses to the c.Dst when needed.
It returns ErrNotControlFrame when given header is not of ms.OpClose, ms.OpPing or ms.OpPong operation code.
func (ControlHandler) HandleClose ¶
func (c ControlHandler) HandleClose(h ms.Header) error
HandleClose handles close frame, makes protocol validity checks and writes specification compatible response to the c.Dst.
func (ControlHandler) HandlePing ¶
func (c ControlHandler) HandlePing(h ms.Header) error
HandlePing handles ping frame and writes specification compatible response to the c.Dst.
func (ControlHandler) HandlePong ¶
func (c ControlHandler) HandlePong(h ms.Header) error
HandlePong handles pong frame by discarding it.
type ControlWriter ¶
type ControlWriter struct {
// contains filtered or unexported fields
}
ControlWriter is a wrapper around Writer that contains some guards for buffered writes of control frames.
func NewControlWriter ¶
NewControlWriter contains ControlWriter with Writer inside whose buffer size is at most ms.MaxControlFramePayloadSize + ms.MaxHeaderSize.
func NewControlWriterBuffer ¶
func NewControlWriterBuffer(dest io.Writer, state ms.State, op ms.OpCode, buf []byte) *ControlWriter
NewControlWriterBuffer returns a new ControlWriter with buf as a buffer.
Note that it reserves x bytes of buf for header data, where x could be ms.MinHeaderSize or ms.MinHeaderSize+4 (depending on state). At most (ms.MaxControlFramePayloadSize + x) bytes of buf will be used.
It panics if len(buf) <= ms.MinHeaderSize + x.
func (*ControlWriter) Flush ¶
func (c *ControlWriter) Flush() error
Flush flushes all buffered data to the underlying io.Writer.
type DebugUpgrader ¶
type DebugUpgrader struct { // Upgrader contains upgrade to WebSocket options. Upgrader ms.Upgrader // OnRequest and OnResponse are the callbacks that will be called with the // HTTP request and response respectively. OnRequest, OnResponse func([]byte) }
DebugUpgrader is a wrapper around ms.Upgrader. It tracks I/O of a WebSocket handshake.
Note that it must not be used in production applications that requires Upgrade() to be efficient.
func (*DebugUpgrader) Upgrade ¶
func (d *DebugUpgrader) Upgrade(conn io.ReadWriter) (hs ms.Handshake, err error)
Upgrade calls Upgrade() on underlying ms.Upgrader and tracks I/O on conn.
type FrameHandlerFunc ¶
FrameHandlerFunc handles parsed frame header and its body represented by io.Reader.
Note that reader represents already unmasked body.
func ControlFrameHandler ¶
func ControlFrameHandler(w io.Writer, state ms.State) FrameHandlerFunc
ControlFrameHandler returns FrameHandlerFunc for handling control frames. For more info see ControlHandler docs.
type Message ¶
Message represents a message from peer, that could be presented in one or more frames. That is, it contains payload of all message fragments and operation code of initial frame for this message.
func ReadClientMessage ¶
ReadClientMessage reads next message from r, considering that caller represents server side. It is a shortcut for ReadMessage(r, ms.StateServerSide, m).
func ReadMessage ¶
ReadMessage is a helper function that reads next message from r. It appends received message(s) to the third argument and returns the result of it and an error if some failure happened. That is, it probably could receive more than one message when peer sending fragmented message in multiple frames and want to send some control frame between fragments. Then returned slice will contain those control frames at first, and then result of gluing fragments.
TODO(gobwas): add DefaultReader with buffer size options.
type Reader ¶
type Reader struct { Source io.Reader State ms.State // SkipHeaderCheck disables checking header bits to be RFC6455 compliant. SkipHeaderCheck bool // CheckUTF8 enables UTF-8 checks for text frames payload. If incoming // bytes are not valid UTF-8 sequence, ErrInvalidUTF8 returned. CheckUTF8 bool // Extensions is a list of negotiated extensions for reader Source. // It is used to meet the specs and clear appropriate bits in fragment // header RSV segment. Extensions []RecvExtension // MaxFrameSize controls the maximum frame size in bytes // that can be read. A message exceeding that size will return // a ErrFrameTooLarge to the application. // // Not setting this field means there is no limit. MaxFrameSize int64 OnContinuation FrameHandlerFunc OnIntermediate FrameHandlerFunc // contains filtered or unexported fields }
Reader is a wrapper around source io.Reader which represents WebSocket connection. It contains options for reading messages from source.
Reader implements io.Reader, which Read() method reads payload of incoming WebSocket frames. It also takes care on fragmented frames and possibly intermediate control frames between them.
Note that Reader's methods are not goroutine safe.
func NewClientSideReader ¶
NewClientSideReader is a helper function that calls NewReader with r and ms.StateClientSide.
func NewReader ¶
NewReader creates new frame reader that reads from r keeping given state to make some protocol validity checks when it needed.
func NewServerSideReader ¶
NewServerSideReader is a helper function that calls NewReader with r and ms.StateServerSide.
func (*Reader) Discard ¶
Discard discards current message unread bytes. It discards all frames of fragmented message.
func (*Reader) NextFrame ¶
NextFrame prepares r to read next message. It returns received frame header and non-nil error on failure.
Note that next NextFrame() call must be done after receiving or discarding all current message bytes.
func (*Reader) Read ¶
Read implements io.Reader. It reads the next message payload into p. It takes care on fragmented messages.
The error is io.EOF only if all of message bytes were read. If an io.EOF happens during reading some but not all the message bytes Read() returns io.ErrUnexpectedEOF.
The error is ErrNoFrameAdvance if no NextFrame() call was made before reading next message bytes.
type RecvExtension ¶
RecvExtension is an interface for clearing fragment header RSV bits.
type RecvExtensionFunc ¶
RecvExtensionFunc is an adapter to allow the use of ordinary functions as RecvExtension.
type SendExtension ¶
SendExtension is an interface for setting fragment header RSV bits.
type SendExtensionFunc ¶
SendExtensionFunc is an adapter to allow the use of ordinary functions as SendExtension.
type UTF8Reader ¶
UTF8Reader implements io.Reader that calculates utf8 validity state after every read byte from Source.
Note that in some cases client must call r.Valid() after all bytes are read to ensure that all of them are valid utf8 sequences. That is, some io helper functions such io.ReadAtLeast or io.ReadFull could discard the error information returned by the reader when they receive all of requested bytes. For example, the last read sequence is invalid and UTF8Reader returns number of bytes read and an error. But helper function decides to discard received error due to all requested bytes are completely read from the source.
Another possible case is when some valid sequence become split by the read bound. Then UTF8Reader can not make decision about validity of the last sequence cause it is not fully read yet. And if the read stops, Valid() will return false, even if Read() by itself dit not.
func NewUTF8Reader ¶
func NewUTF8Reader(r io.Reader) *UTF8Reader
NewUTF8Reader creates utf8 reader that reads from r.
func (*UTF8Reader) Accepted ¶
func (u *UTF8Reader) Accepted() int
Accepted returns number of valid bytes in last Read().
func (*UTF8Reader) Read ¶
func (u *UTF8Reader) Read(p []byte) (n int, err error)
Read implements io.Reader.
func (*UTF8Reader) Reset ¶
func (u *UTF8Reader) Reset(r io.Reader)
Reset resets utf8 reader to read from r.
func (*UTF8Reader) Valid ¶
func (u *UTF8Reader) Valid() bool
Valid checks current reader state. It returns true if all read bytes are valid UTF-8 sequences, and false if not.
type Writer ¶
type Writer struct {
// contains filtered or unexported fields
}
Writer contains logic of buffering output data into a WebSocket fragments. It is much the same as bufio.Writer, except the thing that it works with WebSocket frames, not the raw data.
Writer writes frames with specified OpCode. It uses ms.State to decide whether the output frames must be masked.
Note that it does not check control frame size or other RFC rules. That is, it must be used with special care to write control frames without violation of RFC. You could use ControlWriter that wraps Writer and contains some guards for writing control frames.
If an error occurs writing to a Writer, no more data will be accepted and all subsequent writes will return the error.
After all data has been written, the client should call the Flush() method to guarantee all data has been forwarded to the underlying io.Writer.
func GetWriter ¶
GetWriter tries to reuse Writer getting it from the pool.
This function is intended for memory consumption optimizations, because NewWriter*() functions make allocations for inner buffer.
Note the it ceils n to the power of two.
If you have your own bytes buffer pool you could use NewWriterBuffer to use pooled bytes in writer.
func NewWriterBuffer ¶
NewWriterBuffer returns a new Writer with buf as a buffer.
Note that it reserves x bytes of buf for header data, where x is in range [ms.MinHeaderSize,ms.MaxHeaderSize] (depending on state and buf size).
You could use ms.HeaderSize() to calculate number of bytes needed to store header data.
It panics if len(buf) is too small to fit header and payload data.
func NewWriterBufferSize ¶
NewWriterBufferSize returns a new Writer whose buffer size is equal to n. If n <= ms.MinHeaderSize then the default buffer size is used.
Note that Writer will reserve x bytes for header data, where x is in range [ms.MinHeaderSize,ms.MaxHeaderSize]. That is, frames flushed by Writer will not have payload length equal to n, except the case when Write() is called on empty Writer with len(p) > n.
func NewWriterSize ¶
NewWriterSize returns a new Writer whose buffer size is at most n + ms.MaxHeaderSize. That is, output frames payload length could be up to n, except the case when Write() is called on empty Writer with len(p) > n.
If n <= 0 then the default buffer size is used as Writer's buffer size.
func (*Writer) Buffered ¶
Buffered returns the number of bytes that have been written into the current buffer.
func (*Writer) DisableFlush ¶
func (w *Writer) DisableFlush()
DisableFlush denies Writer to write fragments.
func (*Writer) Flush ¶
Flush writes any buffered data to the underlying io.Writer. It sends the frame with "fin" flag set to true.
If no Write() or ReadFrom() was made, then Flush() does nothing.
func (*Writer) FlushFragment ¶
FlushFragment writes any buffered data to the underlying io.Writer. It sends the frame with "fin" flag set to false.
func (*Writer) Grow ¶
Grow grows Writer's internal buffer capacity to guarantee space for another n bytes of _payload_ -- that is, frame header is not included in n.
func (*Writer) Reset ¶
Reset resets Writer as it was created by New() methods. Note that Reset does reset extensions and other options was set after Writer initialization.
func (*Writer) ResetOp ¶
ResetOp is an quick version of Reset(). ResetOp does reset unwritten fragments and does not reset results of SetExtensions() or DisableFlush() methods.
func (*Writer) SetExtensions ¶
func (w *Writer) SetExtensions(xs ...SendExtension)
SetExtensions adds xs as extensions to be used during writes.
func (*Writer) Size ¶
Size returns the size of the underlying buffer in bytes (not including WebSocket header bytes).