1
0
Fork 0
mirror of https://github.com/diamondburned/arikawa.git synced 2024-11-28 09:42:58 +00:00
arikawa/internal/wsutil/conn.go

180 lines
3.6 KiB
Go

package wsutil
import (
"compress/zlib"
"context"
"io"
"io/ioutil"
"net/http"
"sync"
stderr "errors"
"github.com/diamondburned/arikawa/internal/json"
"github.com/pkg/errors"
"nhooyr.io/websocket"
)
var WSReadLimit int64 = 8192000 // 8 MiB
// Connection is an interface that abstracts around a generic Websocket driver.
// This connection expects the driver to handle compression by itself.
type Connection interface {
// Dial dials the address (string). Context needs to be passed in for
// timeout. This method should also be re-usable after Close is called.
Dial(context.Context, string) error
// Listen sends over events constantly. Error will be non-nil if Data is
// nil, so check for Error first.
Listen() <-chan Event
// Send allows the caller to send bytes. Context needs to be passed in order
// to re-use the context that's already used for the limiter.
Send(context.Context, []byte) error
// Close should close the websocket connection. The connection will not be
// reused.
// If error is nil, the connection should close with a StatusNormalClosure
// (1000). If not, it should close with a StatusProtocolError (1002).
Close(err error) error
}
// Conn is the default Websocket connection. It compresses all payloads using
// zlib.
type Conn struct {
Conn *websocket.Conn
json.Driver
mut sync.Mutex
events chan Event
}
var _ Connection = (*Conn)(nil)
func NewConn(driver json.Driver) *Conn {
return &Conn{
Driver: driver,
events: make(chan Event),
}
}
func (c *Conn) Dial(ctx context.Context, addr string) error {
var err error
headers := http.Header{}
headers.Set("Accept-Encoding", "zlib") // enable
c.mut.Lock()
defer c.mut.Unlock()
c.Conn, _, err = websocket.Dial(ctx, addr, &websocket.DialOptions{
HTTPHeader: headers,
})
if err != nil {
return errors.Wrap(err, "Failed to dial WS")
}
c.Conn.SetReadLimit(WSReadLimit)
c.events = make(chan Event)
c.readLoop()
return err
}
func (c *Conn) Listen() <-chan Event {
return c.events
}
func (c *Conn) readLoop() {
conn := c.Conn
go func() {
defer close(c.events)
for {
b, err := readAll(conn, context.Background())
if err != nil {
// Is the error an EOF?
if stderr.Is(err, io.EOF) {
// Yes it is, exit.
return
}
// Check if the error is a fatal one
if code := websocket.CloseStatus(err); code > -1 {
// Is the exit normal?
if code == websocket.StatusNormalClosure {
return
}
}
// Unusual error; log:
c.events <- Event{nil, errors.Wrap(err, "WS error")}
return
}
c.events <- Event{b, nil}
}
}()
}
func readAll(c *websocket.Conn, ctx context.Context) ([]byte, error) {
t, r, err := c.Reader(ctx)
if err != nil {
return nil, err
}
if t == websocket.MessageBinary {
// Probably a zlib payload
z, err := zlib.NewReader(r)
if err != nil {
c.CloseRead(ctx)
return nil,
errors.Wrap(err, "Failed to create a zlib reader")
}
defer z.Close()
r = z
}
b, err := ioutil.ReadAll(r)
if err != nil {
c.CloseRead(ctx)
return nil, err
}
return b, nil
}
func (c *Conn) Send(ctx context.Context, b []byte) error {
// TODO: zlib stream
return c.Conn.Write(ctx, websocket.MessageText, b)
}
func (c *Conn) Close(err error) error {
// Wait for the read loop to exit after exiting.
defer c.close()
if err == nil {
return c.Conn.Close(websocket.StatusNormalClosure, "")
}
var msg = err.Error()
if len(msg) > 125 {
msg = msg[:125] // truncate
}
return c.Conn.Close(websocket.StatusProtocolError, msg)
}
func (c *Conn) close() {
c.mut.Lock()
defer c.mut.Unlock()
<-c.events
c.events = nil
// Set the connection to nil.
c.Conn = nil
}