state.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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-Encoding", "gzip, deflate, br")
  68. header.Set("Accept-Language", "en-US,en;q=0.7")
  69. header.Set("Origin", "https://discord.com")
  70. header.Set("Priority", "u=0, i")
  71. header.Set("Referer", "https://discord.com/channels/@me")
  72. header.Set("Sec-Fetch-Dest", "empty")
  73. header.Set("Sec-Fetch-Mode", "cors")
  74. header.Set("Sec-Fetch-Site", "same-origin")
  75. return header
  76. }
  77. func onRequest(r httpdriver.Request) error {
  78. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  79. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  80. }
  81. return nil
  82. }
  83. func onRaw(event *ws.RawEvent) {
  84. slog.Debug(
  85. "new raw event",
  86. "code", event.OriginalCode,
  87. "type", event.OriginalType,
  88. // "data", event.Raw,
  89. )
  90. }
  91. func onReadUpdate(event *read.UpdateEvent) {
  92. var guildNode *tview.TreeNode
  93. app.guildsTree.
  94. GetRoot().
  95. Walk(func(node, parent *tview.TreeNode) bool {
  96. switch node.GetReference() {
  97. case event.GuildID:
  98. node.SetTextStyle(app.guildsTree.getGuildNodeStyle(event.GuildID))
  99. guildNode = node
  100. return false
  101. case event.ChannelID:
  102. // private channel
  103. if !event.GuildID.IsValid() {
  104. style := app.guildsTree.getChannelNodeStyle(event.ChannelID)
  105. node.SetTextStyle(style)
  106. return false
  107. }
  108. }
  109. return true
  110. })
  111. if guildNode != nil {
  112. guildNode.Walk(func(node, parent *tview.TreeNode) bool {
  113. if node.GetReference() == event.ChannelID {
  114. node.SetTextStyle(app.guildsTree.getChannelNodeStyle(event.ChannelID))
  115. return false
  116. }
  117. return true
  118. })
  119. }
  120. app.Draw()
  121. }
  122. func onReady(r *gateway.ReadyEvent) {
  123. dmNode := tview.NewTreeNode("Direct Messages")
  124. root := app.guildsTree.
  125. GetRoot().
  126. ClearChildren().
  127. AddChild(dmNode)
  128. for _, folder := range r.UserSettings.GuildFolders {
  129. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  130. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  131. if err != nil {
  132. slog.Error(
  133. "failed to get guild from state",
  134. "guild_id",
  135. folder.GuildIDs[0],
  136. "err",
  137. err,
  138. )
  139. continue
  140. }
  141. app.guildsTree.createGuildNode(root, *guild)
  142. } else {
  143. app.guildsTree.createFolderNode(folder)
  144. }
  145. }
  146. app.guildsTree.SetCurrentNode(root)
  147. app.SetFocus(app.guildsTree)
  148. app.Draw()
  149. }
  150. func onMessageCreate(message *gateway.MessageCreateEvent) {
  151. if app.guildsTree.selectedChannelID == message.ChannelID {
  152. app.messagesList.drawMessage(message.Message)
  153. app.Draw()
  154. }
  155. if err := notifications.Notify(discordState, message, app.cfg); err != nil {
  156. slog.Error("Notification failed", "err", err)
  157. }
  158. }
  159. func onMessageUpdate(message *gateway.MessageUpdateEvent) {
  160. if app.guildsTree.selectedChannelID == message.ChannelID {
  161. onMessageDelete(&gateway.MessageDeleteEvent{ID: message.ID, ChannelID: message.ChannelID, GuildID: message.GuildID})
  162. }
  163. }
  164. func onMessageDelete(message *gateway.MessageDeleteEvent) {
  165. if app.guildsTree.selectedChannelID == message.ChannelID {
  166. messages, err := discordState.Cabinet.Messages(message.ChannelID)
  167. if err != nil {
  168. slog.Error("failed to get messages from state", "err", err, "channel_id", message.ChannelID)
  169. return
  170. }
  171. app.messagesList.reset()
  172. app.messagesList.drawMessages(messages)
  173. app.Draw()
  174. }
  175. }