state.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  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. api.UserAgent = props.BrowserUserAgent
  22. gateway.DefaultIdentity = props
  23. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  24. Status: app.cfg.Identify.Status,
  25. }
  26. discordState = ningen.New(token)
  27. // Handlers
  28. discordState.AddHandler(onRaw)
  29. discordState.AddHandler(onReady)
  30. discordState.AddHandler(onMessageCreate)
  31. discordState.AddHandler(onMessageDelete)
  32. discordState.AddHandler(onReadUpdate)
  33. discordState.AddHandler(func(event *gateway.GuildMembersChunkEvent) {
  34. app.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  35. })
  36. discordState.AddHandler(func(event *gateway.GuildMemberRemoveEvent) {
  37. app.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, discordState.MemberState.SearchLimit)
  38. })
  39. discordState.StateLog = func(err error) {
  40. slog.Error("state log", "err", err)
  41. }
  42. discordState.OnRequest = append(discordState.OnRequest, httputil.WithHeaders(getHeaders(props)), onRequest)
  43. return discordState.Open(context.TODO())
  44. }
  45. func getHeaders(props gateway.IdentifyProperties) http.Header {
  46. header := make(http.Header)
  47. if rawProps, err := json.Marshal(props); err == nil {
  48. propsHeader := base64.StdEncoding.EncodeToString(rawProps)
  49. header.Set("X-Super-Properties", propsHeader)
  50. }
  51. // https://developer.mozilla.org/en-US/docs/Web/HTTP/Reference/Headers
  52. header.Set("Accept", "*/*")
  53. header.Set("Accept-Language", "en-US,en;q=0.7")
  54. header.Set("Origin", "https://discord.com")
  55. header.Set("Referer", "https://discord.com/channels/@me")
  56. header.Set("Sec-Fetch-Dest", "empty")
  57. header.Set("Sec-Fetch-Mode", "cors")
  58. header.Set("Sec-Fetch-Site", "same-origin")
  59. header.Set("X-Debug-Options", "bugReporterEnabled")
  60. header.Set("X-Discord-Locale", string(props.SystemLocale))
  61. return header
  62. }
  63. func onRequest(r httpdriver.Request) error {
  64. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  65. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  66. }
  67. return nil
  68. }
  69. func onRaw(event *ws.RawEvent) {
  70. slog.Debug(
  71. "new raw event",
  72. "code", event.OriginalCode,
  73. "type", event.OriginalType,
  74. "data", event.Raw,
  75. )
  76. }
  77. func onReadUpdate(event *read.UpdateEvent) {
  78. var guildNode *tview.TreeNode
  79. app.guildsTree.
  80. GetRoot().
  81. Walk(func(node, parent *tview.TreeNode) bool {
  82. switch node.GetReference() {
  83. case event.GuildID:
  84. node.SetTextStyle(app.guildsTree.getGuildNodeStyle(event.GuildID))
  85. guildNode = node
  86. return false
  87. case event.ChannelID:
  88. // private channel
  89. if !event.GuildID.IsValid() {
  90. style := app.guildsTree.getChannelNodeStyle(event.ChannelID)
  91. node.SetTextStyle(style)
  92. return false
  93. }
  94. }
  95. return true
  96. })
  97. if guildNode != nil {
  98. guildNode.Walk(func(node, parent *tview.TreeNode) bool {
  99. if node.GetReference() == event.ChannelID {
  100. node.SetTextStyle(app.guildsTree.getChannelNodeStyle(event.ChannelID))
  101. return false
  102. }
  103. return true
  104. })
  105. }
  106. app.Draw()
  107. }
  108. func onReady(r *gateway.ReadyEvent) {
  109. dmNode := tview.NewTreeNode("Direct Messages")
  110. root := app.guildsTree.
  111. GetRoot().
  112. ClearChildren().
  113. AddChild(dmNode)
  114. for _, folder := range r.UserSettings.GuildFolders {
  115. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  116. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  117. if err != nil {
  118. slog.Error(
  119. "failed to get guild from state",
  120. "guild_id",
  121. folder.GuildIDs[0],
  122. "err",
  123. err,
  124. )
  125. continue
  126. }
  127. app.guildsTree.createGuildNode(root, *guild)
  128. } else {
  129. app.guildsTree.createFolderNode(folder)
  130. }
  131. }
  132. app.guildsTree.SetCurrentNode(root)
  133. app.SetFocus(app.guildsTree)
  134. app.Draw()
  135. }
  136. func onMessageCreate(msg *gateway.MessageCreateEvent) {
  137. if app.guildsTree.selectedChannelID.IsValid() &&
  138. app.guildsTree.selectedChannelID == msg.ChannelID {
  139. app.messagesList.createMsg(msg.Message)
  140. app.Draw()
  141. }
  142. if err := notifications.HandleIncomingMessage(discordState, msg, app.cfg); err != nil {
  143. slog.Error("Notification failed", "err", err)
  144. }
  145. }
  146. func onMessageDelete(msg *gateway.MessageDeleteEvent) {
  147. if app.guildsTree.selectedChannelID == msg.ChannelID {
  148. app.messagesList.reset()
  149. app.messagesList.drawMsgs(msg.ChannelID)
  150. app.Draw()
  151. }
  152. }