discord.go 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. package main
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "regexp"
  6. "sort"
  7. "strings"
  8. "github.com/ayntgl/discordgo"
  9. "github.com/gen2brain/beeep"
  10. "github.com/rivo/tview"
  11. )
  12. var (
  13. session *discordgo.Session
  14. selectedChannel *discordgo.Channel
  15. selectedMessage *discordgo.Message
  16. )
  17. var (
  18. boldRegex = regexp.MustCompile(`(?m)\*\*(.*?)\*\*`)
  19. italicRegex = regexp.MustCompile(`(?m)\*(.*?)\*`)
  20. underlineRegex = regexp.MustCompile(`(?m)__(.*?)__`)
  21. strikeThroughRegex = regexp.MustCompile(`(?m)~~(.*?)~~`)
  22. )
  23. func newSession() *discordgo.Session {
  24. s, err := discordgo.New()
  25. if err != nil {
  26. panic(err)
  27. }
  28. s.UserAgent = conf.UserAgent
  29. s.Identify.Compress = false
  30. s.Identify.Intents = 0
  31. s.Identify.LargeThreshold = 0
  32. s.Identify.Properties.Device = ""
  33. s.Identify.Properties.Browser = "Chrome"
  34. s.Identify.Properties.OS = "Linux"
  35. s.AddHandlerOnce(onSessionReady)
  36. s.AddHandler(onSessionMessageCreate)
  37. return s
  38. }
  39. func onSessionReady(_ *discordgo.Session, r *discordgo.Ready) {
  40. dmNode := tview.NewTreeNode("Direct Messages").
  41. Collapse()
  42. n := channelsTree.GetRoot()
  43. n.AddChild(dmNode)
  44. sort.Slice(r.PrivateChannels, func(i, j int) bool {
  45. return r.PrivateChannels[i].LastMessageID > r.PrivateChannels[j].LastMessageID
  46. })
  47. for _, c := range r.PrivateChannels {
  48. var tag string
  49. if isUnread(c) {
  50. tag = "[::b]"
  51. } else {
  52. tag = "[::d]"
  53. }
  54. cn := tview.NewTreeNode(tag + generateChannelRepr(c) + "[::-]").
  55. SetReference(c.ID)
  56. dmNode.AddChild(cn)
  57. }
  58. sort.Slice(r.Guilds, func(a, b int) bool {
  59. found := false
  60. for _, gID := range r.Settings.GuildPositions {
  61. if found {
  62. if gID == r.Guilds[b].ID {
  63. return true
  64. }
  65. } else {
  66. if gID == r.Guilds[a].ID {
  67. found = true
  68. }
  69. }
  70. }
  71. return false
  72. })
  73. for _, g := range r.Guilds {
  74. gn := tview.NewTreeNode(g.Name).Collapse()
  75. n.AddChild(gn)
  76. cs := g.Channels
  77. sort.Slice(cs, func(i, j int) bool {
  78. return cs[i].Position < cs[j].Position
  79. })
  80. // Top-level channels
  81. createTopLevelChannelsTreeNodes(gn, cs)
  82. // Category channels
  83. createCategoryChannelsTreeNodes(gn, cs)
  84. // Second-level channels
  85. createSecondLevelChannelsTreeNodes(cs)
  86. }
  87. channelsTree.SetCurrentNode(n)
  88. }
  89. func isUnread(c *discordgo.Channel) bool {
  90. if c.LastMessageID == "" {
  91. return false
  92. }
  93. for _, rs := range session.State.ReadState {
  94. if c.ID == rs.ID {
  95. return c.LastMessageID != rs.LastMessageID
  96. }
  97. }
  98. return false
  99. }
  100. func onSessionMessageCreate(_ *discordgo.Session, m *discordgo.MessageCreate) {
  101. c, err := session.State.Channel(m.ChannelID)
  102. if err != nil {
  103. return
  104. }
  105. if selectedChannel == nil || selectedChannel.ID != m.ChannelID {
  106. if conf.Notifications {
  107. for _, u := range m.Mentions {
  108. if u.ID == session.State.User.ID {
  109. g, err := session.State.Guild(m.GuildID)
  110. if err != nil {
  111. return
  112. }
  113. go beeep.Alert(fmt.Sprintf("%s (#%s)", g.Name, c.Name), m.ContentWithMentionsReplaced(), "")
  114. return
  115. }
  116. }
  117. }
  118. cn := getTreeNodeByReference(c.ID)
  119. if cn == nil {
  120. return
  121. }
  122. cn.SetText("[::b]" + generateChannelRepr(c) + "[::-]")
  123. app.Draw()
  124. } else {
  125. selectedChannel.Messages = append(selectedChannel.Messages, m.Message)
  126. renderMessage(m.Message)
  127. }
  128. }
  129. type loginResponse struct {
  130. MFA bool `json:"mfa"`
  131. SMS bool `json:"sms"`
  132. Ticket string `json:"ticket"`
  133. Token string `json:"token"`
  134. }
  135. func login(email, password string) (*loginResponse, error) {
  136. data := struct {
  137. Email string `json:"email"`
  138. Password string `json:"password"`
  139. }{email, password}
  140. resp, err := session.RequestWithBucketID(
  141. "POST",
  142. discordgo.EndpointLogin,
  143. data,
  144. discordgo.EndpointLogin,
  145. )
  146. if err != nil {
  147. return nil, err
  148. }
  149. var lr loginResponse
  150. err = json.Unmarshal(resp, &lr)
  151. if err != nil {
  152. return nil, err
  153. }
  154. return &lr, nil
  155. }
  156. func totp(code, ticket string) (*loginResponse, error) {
  157. data := struct {
  158. Code string `json:"code"`
  159. Ticket string `json:"ticket"`
  160. }{code, ticket}
  161. e := discordgo.EndpointAuth + "mfa/totp"
  162. resp, err := session.RequestWithBucketID("POST", e, data, e)
  163. if err != nil {
  164. return nil, err
  165. }
  166. var lr loginResponse
  167. err = json.Unmarshal(resp, &lr)
  168. if err != nil {
  169. return nil, err
  170. }
  171. return &lr, nil
  172. }
  173. func renderMessage(m *discordgo.Message) {
  174. var b strings.Builder
  175. switch m.Type {
  176. case discordgo.MessageTypeDefault, discordgo.MessageTypeReply:
  177. // Define a new region and assign message ID as the region ID.
  178. // Learn more:
  179. // https://pkg.go.dev/github.com/rivo/tview#hdr-Regions_and_Highlights
  180. b.WriteString("[\"")
  181. b.WriteString(m.ID)
  182. b.WriteString("\"]")
  183. // Render the message associated with crosspost, channel follow add,
  184. // pin, or a reply.
  185. if rm := m.ReferencedMessage; rm != nil {
  186. b.WriteString(" ╭ ")
  187. b.WriteString("[::d]")
  188. parseAuthor(&b, rm.Author)
  189. if rm.Content != "" {
  190. rm.Content = parseMentions(rm.Content, rm.Mentions)
  191. b.WriteString(parseMarkdown(rm.Content))
  192. }
  193. b.WriteString("[::-]")
  194. b.WriteByte('\n')
  195. }
  196. // Render the author of the message.
  197. parseAuthor(&b, m.Author)
  198. // If the message content is not empty, parse the message mentions
  199. // (users mentioned in the message) and render the message content.
  200. if m.Content != "" {
  201. m.Content = parseMentions(m.Content, m.Mentions)
  202. b.WriteString(parseMarkdown(m.Content))
  203. }
  204. // If the edited timestamp of the message is not empty; it implies that
  205. // the message has been edited, hence render the message with edited
  206. // label for distinction
  207. if m.EditedTimestamp != "" {
  208. b.WriteString(" [::d](edited)[::-]")
  209. }
  210. // TODO: render message embeds
  211. for range m.Embeds {
  212. b.WriteString("\n<EMBED>")
  213. }
  214. // Render the message attachments (attached files to the message).
  215. for _, a := range m.Attachments {
  216. b.WriteString("\n[")
  217. b.WriteString(a.Filename)
  218. b.WriteString("]: ")
  219. b.WriteString(a.URL)
  220. }
  221. // Tags with no region ID ([""]) do not start new regions. They can
  222. // therefore be used to mark the end of a region.
  223. b.WriteString("[\"\"]")
  224. b.WriteByte('\n')
  225. case discordgo.MessageTypeGuildMemberJoin:
  226. b.WriteString("[#5865F2]")
  227. b.WriteString(m.Author.Username)
  228. b.WriteString("[-] joined the server")
  229. b.WriteByte('\n')
  230. }
  231. if str := b.String(); str != "" {
  232. b := make([]byte, len(str)+1)
  233. copy(b, str)
  234. messagesView.Write(b)
  235. }
  236. }
  237. func parseMentions(content string, mentions []*discordgo.User) string {
  238. for _, mUser := range mentions {
  239. var color string
  240. if mUser.ID == session.State.User.ID {
  241. color = "[:#5865F2]"
  242. } else {
  243. color = "[#EB459E]"
  244. }
  245. content = strings.NewReplacer(
  246. // <@USER_ID>
  247. "<@"+mUser.ID+">",
  248. color+"@"+mUser.Username+"[-:-]",
  249. // <@!USER_ID>
  250. "<@!"+mUser.ID+">",
  251. color+"@"+mUser.Username+"[-:-]",
  252. ).Replace(content)
  253. }
  254. return content
  255. }
  256. func parseAuthor(b *strings.Builder, u *discordgo.User) {
  257. if u.ID == session.State.User.ID {
  258. b.WriteString("[#57F287]")
  259. } else {
  260. b.WriteString("[#ED4245]")
  261. }
  262. b.WriteString(u.Username)
  263. b.WriteString("[-] ")
  264. // If the message author is a bot account, render the message with bot label
  265. // for distinction.
  266. if u.Bot {
  267. b.WriteString("[#EB459E]BOT[-] ")
  268. }
  269. }
  270. func parseMarkdown(md string) string {
  271. var res string
  272. res = boldRegex.ReplaceAllString(md, "[::b]$1[::-]")
  273. res = italicRegex.ReplaceAllString(res, "[::i]$1[::-]")
  274. res = underlineRegex.ReplaceAllString(res, "[::u]$1[::-]")
  275. res = strikeThroughRegex.ReplaceAllString(res, "[::s]$1[::-]")
  276. return res
  277. }