1
0
Fork 0
mirror of https://github.com/diamondburned/arikawa.git synced 2024-11-17 20:32:48 +00:00
arikawa/gateway/op.go

205 lines
4.4 KiB
Go
Raw Normal View History

2020-01-15 04:43:34 +00:00
package gateway
import (
"fmt"
2020-01-18 07:40:44 +00:00
"math/rand"
"time"
2020-01-15 04:43:34 +00:00
"github.com/diamondburned/arikawa/utils/json"
"github.com/diamondburned/arikawa/utils/wsutil"
2020-01-15 04:43:34 +00:00
"github.com/pkg/errors"
)
type OPCode uint8
const (
DispatchOP OPCode = 0 // recv
HeartbeatOP OPCode = 1 // send/recv
IdentifyOP OPCode = 2 // send...
StatusUpdateOP OPCode = 3 //
VoiceStateUpdateOP OPCode = 4 //
VoiceServerPingOP OPCode = 5 //
ResumeOP OPCode = 6 //
ReconnectOP OPCode = 7 // recv
RequestGuildMembersOP OPCode = 8 // send
InvalidSessionOP OPCode = 9 // recv...
HelloOP OPCode = 10
HeartbeatAckOP OPCode = 11
CallConnectOP OPCode = 13
GuildSubscriptionsOP OPCode = 14
2020-01-15 04:43:34 +00:00
)
type OP struct {
Code OPCode `json:"op"`
Data json.Raw `json:"d,omitempty"`
// Only for Dispatch (op 0)
2020-01-15 07:34:18 +00:00
Sequence int64 `json:"s,omitempty"`
2020-01-15 04:43:34 +00:00
EventName string `json:"t,omitempty"`
}
var ErrInvalidSession = errors.New("Invalid session")
func DecodeEvent(driver json.Driver, ev wsutil.Event, v interface{}) (OPCode, error) {
2020-01-15 04:43:34 +00:00
op, err := DecodeOP(driver, ev)
if err != nil {
return 0, err
}
if err := driver.Unmarshal(op.Data, v); err != nil {
return 0, errors.Wrap(err, "Failed to decode data")
}
return op.Code, nil
}
func AssertEvent(driver json.Driver, ev wsutil.Event, code OPCode, v interface{}) (*OP, error) {
2020-01-15 04:43:34 +00:00
op, err := DecodeOP(driver, ev)
if err != nil {
2020-01-15 07:34:18 +00:00
return nil, err
2020-01-15 04:43:34 +00:00
}
if op.Code != code {
2020-01-15 07:34:18 +00:00
return op, fmt.Errorf(
2020-01-15 04:43:34 +00:00
"Unexpected OP Code: %d, expected %d (%s)",
op.Code, code, op.Data,
)
}
if err := driver.Unmarshal(op.Data, v); err != nil {
2020-01-15 07:34:18 +00:00
return op, errors.Wrap(err, "Failed to decode data")
2020-01-15 04:43:34 +00:00
}
2020-01-15 07:34:18 +00:00
return op, nil
2020-01-15 04:43:34 +00:00
}
func HandleEvent(g *Gateway, ev wsutil.Event) error {
o, err := DecodeOP(g.Driver, ev)
if err != nil {
return err
}
return HandleOP(g, o)
}
// WaitForEvent blocks until fn() returns true. All incoming events are handled
// regardless.
func WaitForEvent(g *Gateway, ch <-chan wsutil.Event, fn func(*OP) bool) error {
for ev := range ch {
o, err := DecodeOP(g.Driver, ev)
if err != nil {
return err
}
// Are these events what we're looking for?
found := fn(o)
// Handle the *OP anyway.
if err := HandleOP(g, o); err != nil {
return err
}
// If we found the event, return.
if found {
return nil
}
}
return errors.New("Event not found and event channel is closed.")
}
func DecodeOP(driver json.Driver, ev wsutil.Event) (*OP, error) {
if ev.Error != nil {
return nil, ev.Error
}
if len(ev.Data) == 0 {
return nil, errors.New("Empty payload")
}
2020-01-15 04:43:34 +00:00
var op *OP
if err := driver.Unmarshal(ev.Data, &op); err != nil {
return nil, errors.Wrap(err, "OP error: "+string(ev.Data))
2020-01-15 04:43:34 +00:00
}
if op.Code == InvalidSessionOP {
return op, ErrInvalidSession
}
return op, nil
2020-01-16 03:28:21 +00:00
}
func HandleOP(g *Gateway, op *OP) error {
2020-01-15 04:43:34 +00:00
if g.OP != nil {
g.OP <- op
}
switch op.Code {
case HeartbeatAckOP:
// Heartbeat from the server?
g.Pacemaker.Echo()
case HeartbeatOP:
// Server requesting a heartbeat.
return g.Pacemaker.Pace()
case ReconnectOP:
// Server requests to reconnect, die and retry.
WSDebug("ReconnectOP received.")
// We must reconnect in another goroutine, as running Reconnect
// synchronously would prevent the main event loop from exiting.
go g.Reconnect()
// Gracefully exit with a nil let the event handler take the signal from
// the pacemaker.
return nil
2020-01-15 04:43:34 +00:00
case InvalidSessionOP:
2020-01-18 07:40:44 +00:00
// Discord expects us to sleep for no reason
time.Sleep(time.Duration(rand.Intn(5)+1) * time.Second)
2020-01-15 04:43:34 +00:00
// Invalid session, respond with Identify.
return g.Identify()
case HelloOP:
// What is this OP doing here???
return nil
case DispatchOP:
2020-01-15 07:34:18 +00:00
// Set the sequence
2020-01-16 03:28:21 +00:00
if op.Sequence > 0 {
g.Sequence.Set(op.Sequence)
}
2020-01-15 07:34:18 +00:00
2020-01-15 04:43:34 +00:00
// Check if we know the event
fn, ok := EventCreator[op.EventName]
if !ok {
2020-02-16 05:29:25 +00:00
return fmt.Errorf(
"Unknown event %s: %s",
op.EventName, string(op.Data),
)
2020-01-15 04:43:34 +00:00
}
// Make a new pointer to the event
var ev = fn()
// Try and parse the event
if err := g.Driver.Unmarshal(op.Data, ev); err != nil {
return errors.Wrap(err, "Failed to parse event "+op.EventName)
}
2020-01-16 03:28:21 +00:00
// If the event is a ready, we'll want its sessionID
if ev, ok := ev.(*ReadyEvent); ok {
g.SessionID = ev.SessionID
}
2020-01-15 04:43:34 +00:00
// Throw the event into a channel, it's valid now.
g.Events <- ev
return nil
default:
return fmt.Errorf("Unknown OP code %d (event %s)", op.Code, op.EventName)
2020-01-15 04:43:34 +00:00
}
return nil
}