telegram.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. package integration
  2. import (
  3. "bytes"
  4. "context"
  5. "fmt"
  6. "strconv"
  7. "unicode/utf16"
  8. "github.com/pkg/errors"
  9. apiv1 "github.com/usememos/memos/api/v1"
  10. "github.com/usememos/memos/plugin/telegram"
  11. storepb "github.com/usememos/memos/proto/gen/store"
  12. "github.com/usememos/memos/store"
  13. )
  14. type TelegramHandler struct {
  15. store *store.Store
  16. }
  17. func NewTelegramHandler(store *store.Store) *TelegramHandler {
  18. return &TelegramHandler{store: store}
  19. }
  20. func (t *TelegramHandler) BotToken(ctx context.Context) string {
  21. return t.store.GetSystemSettingValueWithDefault(ctx, apiv1.SystemSettingTelegramBotTokenName.String(), "")
  22. }
  23. const (
  24. workingMessage = "Working on sending your memo..."
  25. successMessage = "Success"
  26. )
  27. func (t *TelegramHandler) MessageHandle(ctx context.Context, bot *telegram.Bot, message telegram.Message, attachments []telegram.Attachment) error {
  28. reply, err := bot.SendReplyMessage(ctx, message.Chat.ID, message.MessageID, workingMessage)
  29. if err != nil {
  30. return errors.Wrap(err, "Failed to SendReplyMessage")
  31. }
  32. var creatorID int32
  33. userSettingList, err := t.store.ListUserSettingsV1(ctx, &store.FindUserSetting{
  34. Key: storepb.UserSettingKey_USER_SETTING_TELEGRAM_USER_ID,
  35. })
  36. if err != nil {
  37. return errors.Wrap(err, "Failed to find userSettingList")
  38. }
  39. for _, userSetting := range userSettingList {
  40. if userSetting.GetTelegramUserId() == strconv.FormatInt(message.From.ID, 10) {
  41. creatorID = userSetting.UserId
  42. }
  43. }
  44. if creatorID == 0 {
  45. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Please set your telegram userid %d in UserSetting of memos", message.From.ID), nil)
  46. return err
  47. }
  48. create := &store.Memo{
  49. CreatorID: creatorID,
  50. Visibility: store.Private,
  51. }
  52. if message.Text != nil {
  53. create.Content = convertToMarkdown(*message.Text, message.Entities)
  54. }
  55. if message.Caption != nil {
  56. create.Content = convertToMarkdown(*message.Caption, message.CaptionEntities)
  57. }
  58. if message.ForwardFromChat != nil {
  59. create.Content += fmt.Sprintf("\n\n[Message link](%s)", message.GetMessageLink())
  60. }
  61. memoMessage, err := t.store.CreateMemo(ctx, create)
  62. if err != nil {
  63. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to CreateMemo: %s", err), nil)
  64. return err
  65. }
  66. // create resources
  67. for _, attachment := range attachments {
  68. // Fill the common field of create
  69. create := store.Resource{
  70. CreatorID: creatorID,
  71. Filename: attachment.FileName,
  72. Type: attachment.GetMimeType(),
  73. Size: attachment.FileSize,
  74. MemoID: &memoMessage.ID,
  75. }
  76. err := apiv1.SaveResourceBlob(ctx, t.store, &create, bytes.NewReader(attachment.Data))
  77. if err != nil {
  78. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to SaveResourceBlob: %s", err), nil)
  79. return err
  80. }
  81. _, err = t.store.CreateResource(ctx, &create)
  82. if err != nil {
  83. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to CreateResource: %s", err), nil)
  84. return err
  85. }
  86. }
  87. keyboard := generateKeyboardForMemoID(memoMessage.ID)
  88. _, err = bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Saved as %s Memo %d", memoMessage.Visibility, memoMessage.ID), keyboard)
  89. return err
  90. }
  91. func (t *TelegramHandler) CallbackQueryHandle(ctx context.Context, bot *telegram.Bot, callbackQuery telegram.CallbackQuery) error {
  92. var memoID int32
  93. var visibility store.Visibility
  94. n, err := fmt.Sscanf(callbackQuery.Data, "%s %d", &visibility, &memoID)
  95. if err != nil || n != 2 {
  96. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to parse callbackQuery.Data %s", callbackQuery.Data))
  97. }
  98. update := store.UpdateMemo{
  99. ID: memoID,
  100. Visibility: &visibility,
  101. }
  102. err = t.store.UpdateMemo(ctx, &update)
  103. if err != nil {
  104. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to call UpdateMemo %s", err))
  105. }
  106. keyboard := generateKeyboardForMemoID(memoID)
  107. _, err = bot.EditMessage(ctx, callbackQuery.Message.Chat.ID, callbackQuery.Message.MessageID, fmt.Sprintf("Saved as %s Memo %d", visibility, memoID), keyboard)
  108. if err != nil {
  109. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to EditMessage %s", err))
  110. }
  111. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Success changing Memo %d to %s", memoID, visibility))
  112. }
  113. func generateKeyboardForMemoID(id int32) [][]telegram.InlineKeyboardButton {
  114. allVisibility := []store.Visibility{
  115. store.Public,
  116. store.Protected,
  117. store.Private,
  118. }
  119. buttons := make([]telegram.InlineKeyboardButton, 0, len(allVisibility))
  120. for _, v := range allVisibility {
  121. button := telegram.InlineKeyboardButton{
  122. Text: v.String(),
  123. CallbackData: fmt.Sprintf("%s %d", v, id),
  124. }
  125. buttons = append(buttons, button)
  126. }
  127. return [][]telegram.InlineKeyboardButton{buttons}
  128. }
  129. func convertToMarkdown(text string, messageEntities []telegram.MessageEntity) string {
  130. insertions := make(map[int]string)
  131. for _, e := range messageEntities {
  132. var before, after string
  133. // this is supported by the current markdown
  134. switch e.Type {
  135. case telegram.Bold:
  136. before = "**"
  137. after = "**"
  138. case telegram.Italic:
  139. before = "*"
  140. after = "*"
  141. case telegram.Strikethrough:
  142. before = "~~"
  143. after = "~~"
  144. case telegram.Code:
  145. before = "`"
  146. after = "`"
  147. case telegram.Pre:
  148. before = "```" + e.Language
  149. after = "```"
  150. case telegram.TextLink:
  151. before = "["
  152. after = fmt.Sprintf(`](%s)`, e.URL)
  153. }
  154. if before != "" {
  155. insertions[e.Offset] += before
  156. insertions[e.Offset+e.Length] = after + insertions[e.Offset+e.Length]
  157. }
  158. }
  159. input := []rune(text)
  160. var output []rune
  161. utf16pos := 0
  162. for i := 0; i < len(input); i++ {
  163. output = append(output, []rune(insertions[utf16pos])...)
  164. output = append(output, input[i])
  165. utf16pos += len(utf16.Encode([]rune{input[i]}))
  166. }
  167. output = append(output, []rune(insertions[utf16pos])...)
  168. return string(output)
  169. }