state.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. package cmd
  2. import (
  3. "context"
  4. "encoding/base64"
  5. "encoding/json"
  6. "log/slog"
  7. "net/http"
  8. "github.com/ayn2op/discordo/internal/consts"
  9. "github.com/ayn2op/discordo/internal/notifications"
  10. "github.com/ayn2op/tview"
  11. "github.com/diamondburned/arikawa/v3/api"
  12. "github.com/diamondburned/arikawa/v3/gateway"
  13. "github.com/diamondburned/arikawa/v3/utils/httputil"
  14. "github.com/diamondburned/arikawa/v3/utils/httputil/httpdriver"
  15. "github.com/diamondburned/arikawa/v3/utils/ws"
  16. "github.com/diamondburned/ningen/v3"
  17. "github.com/diamondburned/ningen/v3/states/read"
  18. )
  19. func openState(token string) error {
  20. props := consts.GetIdentifyProps()
  21. if browserUserAgent, ok := props["browser_user_agent"]; ok {
  22. if val, ok := browserUserAgent.(string); ok {
  23. api.UserAgent = val
  24. }
  25. }
  26. gateway.DefaultIdentity = props
  27. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  28. Status: app.cfg.Status,
  29. }
  30. discordState = ningen.New(token)
  31. // Handlers
  32. discordState.AddHandler(onRaw)
  33. discordState.AddHandler(onReady)
  34. discordState.AddHandler(onMessageCreate)
  35. discordState.AddHandler(onMessageUpdate)
  36. discordState.AddHandler(onMessageDelete)
  37. discordState.AddHandler(onReadUpdate)
  38. discordState.AddHandler(func(event *gateway.GuildMembersChunkEvent) {
  39. app.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  40. })
  41. discordState.AddHandler(func(event *gateway.GuildMemberRemoveEvent) {
  42. app.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, discordState.MemberState.SearchLimit)
  43. })
  44. discordState.StateLog = func(err error) {
  45. slog.Error("state log", "err", err)
  46. }
  47. discordState.OnRequest = append(discordState.OnRequest, httputil.WithHeaders(getHeaders(props)), onRequest)
  48. return discordState.Open(context.TODO())
  49. }
  50. func getHeaders(props gateway.IdentifyProperties) http.Header {
  51. header := make(http.Header)
  52. // These properties are only sent when identifying with the gateway and are not included in the X-Super-Properties header.
  53. delete(props, "is_fast_connect")
  54. delete(props, "gateway_connect_reasons")
  55. if rawProps, err := json.Marshal(props); err == nil {
  56. propsHeader := base64.StdEncoding.EncodeToString(rawProps)
  57. header.Set("X-Super-Properties", propsHeader)
  58. }
  59. if systemLocale, ok := props["system_locale"]; ok {
  60. if val, ok := systemLocale.(string); ok {
  61. header.Set("X-Discord-Locale", string(val))
  62. }
  63. }
  64. header.Set("X-Debug-Options", "bugReporterEnabled")
  65. // https://developer.mozilla.org/en-US/docs/Web/HTTP/Reference/Headers
  66. header.Set("Accept", "*/*")
  67. header.Set("Accept-Language", "en-US,en;q=0.7")
  68. header.Set("Origin", "https://discord.com")
  69. header.Set("Referer", "https://discord.com/channels/@me")
  70. header.Set("Sec-Fetch-Dest", "empty")
  71. header.Set("Sec-Fetch-Mode", "cors")
  72. header.Set("Sec-Fetch-Site", "same-origin")
  73. return header
  74. }
  75. func onRequest(r httpdriver.Request) error {
  76. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  77. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  78. }
  79. return nil
  80. }
  81. func onRaw(event *ws.RawEvent) {
  82. slog.Debug(
  83. "new raw event",
  84. "code", event.OriginalCode,
  85. "type", event.OriginalType,
  86. // "data", event.Raw,
  87. )
  88. }
  89. func onReadUpdate(event *read.UpdateEvent) {
  90. var guildNode *tview.TreeNode
  91. app.guildsTree.
  92. GetRoot().
  93. Walk(func(node, parent *tview.TreeNode) bool {
  94. switch node.GetReference() {
  95. case event.GuildID:
  96. node.SetTextStyle(app.guildsTree.getGuildNodeStyle(event.GuildID))
  97. guildNode = node
  98. return false
  99. case event.ChannelID:
  100. // private channel
  101. if !event.GuildID.IsValid() {
  102. style := app.guildsTree.getChannelNodeStyle(event.ChannelID)
  103. node.SetTextStyle(style)
  104. return false
  105. }
  106. }
  107. return true
  108. })
  109. if guildNode != nil {
  110. guildNode.Walk(func(node, parent *tview.TreeNode) bool {
  111. if node.GetReference() == event.ChannelID {
  112. node.SetTextStyle(app.guildsTree.getChannelNodeStyle(event.ChannelID))
  113. return false
  114. }
  115. return true
  116. })
  117. }
  118. app.Draw()
  119. }
  120. func onReady(r *gateway.ReadyEvent) {
  121. dmNode := tview.NewTreeNode("Direct Messages")
  122. root := app.guildsTree.
  123. GetRoot().
  124. ClearChildren().
  125. AddChild(dmNode)
  126. for _, folder := range r.UserSettings.GuildFolders {
  127. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  128. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  129. if err != nil {
  130. slog.Error(
  131. "failed to get guild from state",
  132. "guild_id",
  133. folder.GuildIDs[0],
  134. "err",
  135. err,
  136. )
  137. continue
  138. }
  139. app.guildsTree.createGuildNode(root, *guild)
  140. } else {
  141. app.guildsTree.createFolderNode(folder)
  142. }
  143. }
  144. app.guildsTree.SetCurrentNode(root)
  145. app.SetFocus(app.guildsTree)
  146. app.Draw()
  147. }
  148. func onMessageCreate(message *gateway.MessageCreateEvent) {
  149. if app.guildsTree.selectedChannelID == message.ChannelID {
  150. app.messagesList.drawMessage(message.Message)
  151. app.Draw()
  152. }
  153. if err := notifications.Notify(discordState, message, app.cfg); err != nil {
  154. slog.Error("Notification failed", "err", err)
  155. }
  156. }
  157. func onMessageUpdate(message *gateway.MessageUpdateEvent) {
  158. if app.guildsTree.selectedChannelID == message.ChannelID {
  159. onMessageDelete(&gateway.MessageDeleteEvent{ID: message.ID, ChannelID: message.ChannelID, GuildID: message.GuildID})
  160. }
  161. }
  162. func onMessageDelete(message *gateway.MessageDeleteEvent) {
  163. if app.guildsTree.selectedChannelID == message.ChannelID {
  164. messages, err := discordState.Cabinet.Messages(message.ChannelID)
  165. if err != nil {
  166. slog.Error("failed to get messages from state", "err", err, "channel_id", message.ChannelID)
  167. return
  168. }
  169. app.messagesList.reset()
  170. app.messagesList.drawMessages(messages)
  171. app.Draw()
  172. }
  173. }