mirror of
https://github.com/diamondburned/arikawa.git
synced 2024-11-09 16:35:12 +00:00
203 lines
4.5 KiB
Go
203 lines
4.5 KiB
Go
package gateway
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"time"
|
|
|
|
"github.com/diamondburned/arikawa/utils/json"
|
|
"github.com/diamondburned/arikawa/utils/wsutil"
|
|
"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
|
|
)
|
|
|
|
type OP struct {
|
|
Code OPCode `json:"op"`
|
|
Data json.Raw `json:"d,omitempty"`
|
|
|
|
// Only for Dispatch (op 0)
|
|
Sequence int64 `json:"s,omitempty"`
|
|
EventName string `json:"t,omitempty"`
|
|
}
|
|
|
|
func DecodeEvent(driver json.Driver, ev wsutil.Event, v interface{}) (OPCode, error) {
|
|
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) {
|
|
op, err := DecodeOP(driver, ev)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if op.Code != code {
|
|
return op, fmt.Errorf(
|
|
"Unexpected OP Code: %d, expected %d (%s)",
|
|
op.Code, code, op.Data,
|
|
)
|
|
}
|
|
|
|
if err := driver.Unmarshal(op.Data, v); err != nil {
|
|
return op, errors.Wrap(err, "Failed to decode data")
|
|
}
|
|
|
|
return op, nil
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
// Handle the *OP first, in case it's an Invalid Session. This should
|
|
// also prevent a race condition with things that need Ready after
|
|
// Open().
|
|
if err := HandleOP(g, o); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Are these events what we're looking for? If we've found the event,
|
|
// return.
|
|
if fn(o) {
|
|
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")
|
|
}
|
|
|
|
var op *OP
|
|
if err := driver.Unmarshal(ev.Data, &op); err != nil {
|
|
return nil, errors.Wrap(err, "OP error: "+string(ev.Data))
|
|
}
|
|
|
|
return op, nil
|
|
}
|
|
|
|
func HandleOP(g *Gateway, op *OP) error {
|
|
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
|
|
|
|
case InvalidSessionOP:
|
|
// Discord expects us to sleep for no reason
|
|
time.Sleep(time.Duration(rand.Intn(5)+1) * time.Second)
|
|
|
|
// Invalid session, try and Identify.
|
|
if err := g.Identify(); err != nil {
|
|
// Can't identify, reconnect.
|
|
go g.Reconnect()
|
|
}
|
|
return nil
|
|
|
|
case HelloOP:
|
|
// What is this OP doing here???
|
|
return nil
|
|
|
|
case DispatchOP:
|
|
// Set the sequence
|
|
if op.Sequence > 0 {
|
|
g.Sequence.Set(op.Sequence)
|
|
}
|
|
|
|
// Check if we know the event
|
|
fn, ok := EventCreator[op.EventName]
|
|
if !ok {
|
|
return fmt.Errorf(
|
|
"Unknown event %s: %s",
|
|
op.EventName, string(op.Data),
|
|
)
|
|
}
|
|
|
|
// 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)
|
|
}
|
|
|
|
// If the event is a ready, we'll want its sessionID
|
|
if ev, ok := ev.(*ReadyEvent); ok {
|
|
g.SessionID = ev.SessionID
|
|
}
|
|
|
|
// 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)
|
|
}
|
|
|
|
return nil
|
|
}
|