arikawa/internal/wsutil/conn.go

149 lines
3.2 KiB
Go
Raw Normal View History

2020-01-09 05:24:45 +00:00
package wsutil
import (
"compress/zlib"
"context"
"io/ioutil"
"net/http"
2020-01-15 18:32:54 +00:00
"github.com/diamondburned/arikawa/internal/json"
2020-01-09 05:24:45 +00:00
"github.com/pkg/errors"
"nhooyr.io/websocket"
)
var WSBuffer = 12
var WSReadLimit int64 = 8192000 // 8 MiB
2020-01-09 05:24:45 +00:00
// Connection is an interface that abstracts around a generic Websocket driver.
// This connection expects the driver to handle compression by itself.
type Connection interface {
2020-01-15 04:43:34 +00:00
// 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
2020-01-09 05:24:45 +00:00
// 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 {
*websocket.Conn
json.Driver
events chan Event
}
var _ Connection = (*Conn)(nil)
func NewConn(driver json.Driver) *Conn {
return &Conn{
2020-01-17 22:29:13 +00:00
Driver: driver,
events: make(chan Event, WSBuffer),
2020-01-09 05:24:45 +00:00
}
}
func (c *Conn) Dial(ctx context.Context, addr string) error {
var err error
headers := http.Header{}
headers.Set("Accept-Encoding", "zlib") // enable
c.Conn, _, err = websocket.Dial(ctx, addr, &websocket.DialOptions{
HTTPHeader: headers,
})
c.Conn.SetReadLimit(WSReadLimit)
2020-01-16 03:28:21 +00:00
go func() {
c.readLoop(c.events)
}()
2020-01-09 05:24:45 +00:00
return err
}
func (c *Conn) Listen() <-chan Event {
return c.events
}
func (c *Conn) readLoop(ch chan Event) {
for {
2020-01-17 22:29:13 +00:00
b, err := c.readAll(context.Background())
2020-01-09 05:24:45 +00:00
if err != nil {
// Check if the error is a fatal one
2020-01-16 03:28:21 +00:00
if code := websocket.CloseStatus(err); code > -1 {
// Is the exit unusual?
if code != websocket.StatusNormalClosure {
// Unusual error, log
ch <- Event{nil, errors.Wrap(err, "WS fatal")}
}
2020-01-09 05:24:45 +00:00
return
}
2020-01-16 03:28:21 +00:00
// or it's not fatal, we just log and continue
ch <- Event{nil, errors.Wrap(err, "WS error")}
2020-01-09 05:24:45 +00:00
continue
}
ch <- Event{b, nil}
}
}
func (c *Conn) readAll(ctx context.Context) ([]byte, error) {
t, r, err := c.Reader(ctx)
if err != nil {
2020-01-16 03:28:21 +00:00
return nil, err
2020-01-09 05:24:45 +00:00
}
if t == websocket.MessageBinary {
// Probably a zlib payload
z, err := zlib.NewReader(r)
if err != nil {
c.CloseRead(ctx)
2020-01-09 05:24:45 +00:00
return nil,
errors.Wrap(err, "Failed to create a zlib reader")
}
defer z.Close()
r = z
}
b, err := ioutil.ReadAll(r)
2020-01-09 05:24:45 +00:00
if err != nil {
c.CloseRead(ctx)
return nil, err
2020-01-09 05:24:45 +00:00
}
return b, nil
}
2020-01-09 05:24:45 +00:00
func (c *Conn) Send(ctx context.Context, b []byte) error {
// TODO: zlib stream
return c.Write(ctx, websocket.MessageText, b)
2020-01-09 05:24:45 +00:00
}
func (c *Conn) Close(err error) error {
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)
}