cchat-discord/internal/_discord/session/session.go

157 lines
3.9 KiB
Go
Raw Normal View History

2020-09-08 04:44:09 +00:00
package session
import (
"context"
2020-12-20 05:44:26 +00:00
"github.com/diamondburned/arikawa/v2/gateway"
"github.com/diamondburned/arikawa/v2/session"
2020-09-08 04:44:09 +00:00
"github.com/diamondburned/cchat"
"github.com/diamondburned/cchat-discord/internal/discord/folder"
"github.com/diamondburned/cchat-discord/internal/discord/guild"
"github.com/diamondburned/cchat-discord/internal/discord/private"
2020-09-08 04:44:09 +00:00
"github.com/diamondburned/cchat-discord/internal/discord/state"
"github.com/diamondburned/cchat-discord/internal/urlutils"
"github.com/diamondburned/cchat/text"
2020-10-07 01:53:15 +00:00
"github.com/diamondburned/cchat/utils/empty"
2020-12-20 05:44:26 +00:00
"github.com/diamondburned/ningen/v2"
2020-09-08 04:44:09 +00:00
"github.com/pkg/errors"
)
2020-10-07 01:53:15 +00:00
var ErrMFA = session.ErrMFA
2020-09-08 04:44:09 +00:00
type Session struct {
empty.Session
private cchat.Server
state *state.Instance
2020-09-08 04:44:09 +00:00
}
2020-10-07 01:53:15 +00:00
func NewFromInstance(i *state.Instance) (cchat.Session, error) {
priv, err := private.New(i)
if err != nil {
return nil, errors.Wrap(err, "failed to make main private server")
}
return &Session{
private: priv,
state: i,
}, nil
2020-09-08 04:44:09 +00:00
}
func (s *Session) ID() cchat.ID {
return s.state.UserID.String()
2020-09-08 04:44:09 +00:00
}
func (s *Session) Name() text.Rich {
2020-12-20 05:44:26 +00:00
u, err := s.state.Cabinet.Me()
2020-09-08 04:44:09 +00:00
if err != nil {
// This shouldn't happen, ever.
return text.Rich{Content: "<@" + s.state.UserID.String() + ">"}
2020-09-08 04:44:09 +00:00
}
return text.Rich{Content: u.Username + "#" + u.Discriminator}
}
2020-10-07 01:53:15 +00:00
func (s *Session) AsIconer() cchat.Iconer { return s }
2020-09-08 04:44:09 +00:00
func (s *Session) Icon(ctx context.Context, iconer cchat.IconContainer) (func(), error) {
u, err := s.state.Me()
2020-09-08 04:44:09 +00:00
if err != nil {
return nil, errors.Wrap(err, "Failed to get the current user")
}
// Thanks to arikawa, AvatarURL is never empty.
iconer.SetIcon(urlutils.AvatarURL(u.AvatarURL()))
return s.state.AddHandler(func(*gateway.UserUpdateEvent) {
2020-09-08 04:44:09 +00:00
// Bypass the event and use the state cache.
2020-12-20 05:44:26 +00:00
if u, err := s.state.Cabinet.Me(); err == nil {
2020-09-08 04:44:09 +00:00
iconer.SetIcon(urlutils.AvatarURL(u.AvatarURL()))
}
}), nil
}
func (s *Session) Disconnect() error {
return s.state.Close()
2020-09-08 04:44:09 +00:00
}
func (s *Session) AsSessionSaver() cchat.SessionSaver { return s.state }
2020-09-08 04:44:09 +00:00
func (s *Session) Servers(container cchat.ServersContainer) error {
// Reset the entire container when the session is closed.
s.state.AddHandler(func(*session.Closed) {
2020-09-08 04:44:09 +00:00
container.SetServers(nil)
})
// Set the entire container again once reconnected.
s.state.AddHandler(func(*ningen.Connected) {
2021-01-03 04:44:30 +00:00
s.servers(container)
2020-09-08 04:44:09 +00:00
})
return s.servers(container)
}
func (s *Session) servers(container cchat.ServersContainer) error {
2020-12-20 05:44:26 +00:00
ready := s.state.Ready()
2020-09-08 04:44:09 +00:00
// If the user has guild folders:
2021-03-13 08:21:12 +00:00
if len(ready.UserSettings.GuildFolders) > 0 {
2020-09-08 04:44:09 +00:00
// TODO: account for missing guilds.
2020-12-20 05:44:26 +00:00
toplevels := make([]cchat.Server, 1, len(ready.UserSettings.GuildFolders)+1)
toplevels[0] = s.private
2020-09-08 04:44:09 +00:00
2020-12-20 05:44:26 +00:00
for _, guildFolder := range ready.UserSettings.GuildFolders {
2020-09-08 04:44:09 +00:00
// TODO: correct.
switch {
case guildFolder.ID != 0:
2020-09-08 04:44:09 +00:00
fallthrough
case len(guildFolder.GuildIDs) > 1:
toplevels = append(toplevels, folder.New(s.state, guildFolder))
2020-09-08 04:44:09 +00:00
case len(guildFolder.GuildIDs) == 1:
g, err := guild.NewFromID(s.state, guildFolder.GuildIDs[0])
2020-09-08 04:44:09 +00:00
if err != nil {
continue
}
toplevels = append(toplevels, g)
}
}
container.SetServers(toplevels)
2021-03-13 08:21:12 +00:00
return nil
}
2020-09-08 04:44:09 +00:00
// If the user doesn't have guild folders but has sorted their guilds
// before:
2021-03-13 08:21:12 +00:00
if len(ready.UserSettings.GuildPositions) > 0 {
2020-12-20 05:44:26 +00:00
guilds := make([]cchat.Server, 1, len(ready.UserSettings.GuildPositions)+1)
guilds[0] = s.private
2020-09-08 04:44:09 +00:00
2020-12-20 05:44:26 +00:00
for _, id := range ready.UserSettings.GuildPositions {
g, err := guild.NewFromID(s.state, id)
2020-09-08 04:44:09 +00:00
if err != nil {
continue
}
guilds = append(guilds, g)
}
container.SetServers(guilds)
2021-03-13 08:21:12 +00:00
return nil
}
2020-09-08 04:44:09 +00:00
// None of the above:
2021-03-13 08:21:12 +00:00
g, err := s.state.Guilds()
if err != nil {
return err
}
2021-03-13 08:21:12 +00:00
servers := make([]cchat.Server, len(g)+1)
servers[0] = s.private
2020-09-08 04:44:09 +00:00
2021-03-13 08:21:12 +00:00
for i := range g {
servers[i+1] = guild.New(s.state, &g[i])
2020-09-08 04:44:09 +00:00
}
2021-03-13 08:21:12 +00:00
container.SetServers(servers)
2020-09-08 04:44:09 +00:00
return nil
}