1
0
Fork 0
mirror of https://github.com/diamondburned/arikawa.git synced 2025-01-23 21:16:42 +00:00
arikawa/voice/voice.go

188 lines
4.3 KiB
Go
Raw Normal View History

2020-04-25 03:29:14 +00:00
// Package voice handles the Discord voice gateway and UDP connections, as well
// as managing and keeping track of multiple voice sessions.
//
// This package abstracts the subpackage voice/voicesession and voice/udp.
2020-04-20 01:21:10 +00:00
package voice
import (
"log"
"strconv"
2020-04-20 01:21:10 +00:00
"sync"
"github.com/diamondburned/arikawa/discord"
"github.com/diamondburned/arikawa/gateway"
"github.com/diamondburned/arikawa/state"
"github.com/pkg/errors"
)
var (
// defaultErrorHandler is the default error handler
defaultErrorHandler = func(err error) { log.Println("Voice gateway error:", err) }
// ErrCannotSend is an error when audio is sent to a closed channel.
ErrCannotSend = errors.New("cannot send audio to closed channel")
2020-04-20 01:21:10 +00:00
)
// Voice represents a Voice Repository used for managing voice sessions.
2020-04-20 01:21:10 +00:00
type Voice struct {
*state.State
2020-04-20 01:21:10 +00:00
// Session holds all of the active voice sessions.
mapmutex sync.Mutex
sessions map[discord.Snowflake]*Session // guildID:Session
2020-04-20 01:21:10 +00:00
// ErrorLog will be called when an error occurs (defaults to log.Println)
ErrorLog func(err error)
}
// NewVoice creates a new Voice repository wrapped around a state.
2020-04-20 01:21:10 +00:00
func NewVoice(s *state.State) *Voice {
v := &Voice{
State: s,
sessions: make(map[discord.Snowflake]*Session),
2020-04-20 01:21:10 +00:00
ErrorLog: defaultErrorHandler,
}
// Add the required event handlers to the session.
s.AddHandler(v.onVoiceStateUpdate)
s.AddHandler(v.onVoiceServerUpdate)
return v
}
// onVoiceStateUpdate receives VoiceStateUpdateEvents from the gateway
// to keep track of the current user's voice state.
func (v *Voice) onVoiceStateUpdate(e *gateway.VoiceStateUpdateEvent) {
// Get the current user.
me, err := v.Me()
if err != nil {
v.ErrorLog(err)
return
}
2020-04-20 01:21:10 +00:00
// Ignore the event if it is an update from another user.
if me.ID != e.UserID {
return
}
2020-04-20 01:21:10 +00:00
// Get the stored voice session for the given guild.
vs, ok := v.GetSession(e.GuildID)
if !ok {
return
}
2020-04-20 01:21:10 +00:00
// Do what we must.
vs.UpdateState(e)
2020-04-20 01:21:10 +00:00
// Remove the connection if the current user has disconnected.
if e.ChannelID == 0 {
v.RemoveSession(e.GuildID)
}
}
2020-04-20 01:21:10 +00:00
// onVoiceServerUpdate receives VoiceServerUpdateEvents from the gateway
// to manage the current user's voice connections.
func (v *Voice) onVoiceServerUpdate(e *gateway.VoiceServerUpdateEvent) {
// Get the stored voice session for the given guild.
vs, ok := v.GetSession(e.GuildID)
2020-04-20 01:21:10 +00:00
if !ok {
return
2020-04-20 01:21:10 +00:00
}
// Do what we must.
vs.UpdateServer(e)
}
2020-04-20 01:21:10 +00:00
// GetSession gets a session for a guild with a read lock.
func (v *Voice) GetSession(guildID discord.Snowflake) (*Session, bool) {
v.mapmutex.Lock()
defer v.mapmutex.Unlock()
2020-04-20 01:21:10 +00:00
// For some reason you cannot just put `return v.sessions[]` and return a bool D:
conn, ok := v.sessions[guildID]
return conn, ok
}
2020-04-20 01:21:10 +00:00
// RemoveSession removes a session.
func (v *Voice) RemoveSession(guildID discord.Snowflake) {
v.mapmutex.Lock()
defer v.mapmutex.Unlock()
// Ensure that the session is disconnected.
if ses, ok := v.sessions[guildID]; ok {
ses.Disconnect()
2020-04-20 01:21:10 +00:00
}
delete(v.sessions, guildID)
}
2020-04-20 01:21:10 +00:00
// JoinChannel joins the specified channel in the specified guild.
func (v *Voice) JoinChannel(gID, cID discord.Snowflake, muted, deafened bool) (*Session, error) {
// Get the stored voice session for the given guild.
conn, ok := v.GetSession(gID)
2020-04-20 01:21:10 +00:00
// Create a new voice session if one does not exist.
if !ok {
u, err := v.Me()
if err != nil {
return nil, errors.Wrap(err, "Failed to get self")
}
2020-04-20 01:21:10 +00:00
conn = NewSession(v.Session, u.ID)
v.mapmutex.Lock()
v.sessions[gID] = conn
v.mapmutex.Unlock()
}
2020-04-20 01:21:10 +00:00
// Connect.
return conn, conn.JoinChannel(gID, cID, muted, deafened)
2020-04-20 01:21:10 +00:00
}
type CloseError struct {
SessionErrors map[discord.Snowflake]error
StateErr error
}
func (e *CloseError) HasError() bool {
if e.StateErr != nil {
return true
}
return len(e.SessionErrors) > 0
}
func (e *CloseError) Error() string {
if e.StateErr != nil {
return e.StateErr.Error()
}
if len(e.SessionErrors) < 1 {
return ""
}
return strconv.Itoa(len(e.SessionErrors)) + " voice sessions returned errors while attempting to disconnect"
}
func (v *Voice) Close() error {
err := &CloseError{
SessionErrors: make(map[discord.Snowflake]error),
}
v.mapmutex.Lock()
defer v.mapmutex.Unlock()
for gID, s := range v.sessions {
if dErr := s.Disconnect(); dErr != nil {
err.SessionErrors[gID] = dErr
}
}
err.StateErr = v.State.Close()
if err.HasError() {
return err
}
return nil
}