telegram.go 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. package integration
  2. import (
  3. "bytes"
  4. "context"
  5. "fmt"
  6. "path/filepath"
  7. "strconv"
  8. "time"
  9. "unicode/utf16"
  10. "github.com/lithammer/shortuuid/v4"
  11. "github.com/pkg/errors"
  12. apiv1 "github.com/usememos/memos/api/v1"
  13. "github.com/usememos/memos/plugin/telegram"
  14. "github.com/usememos/memos/plugin/webhook"
  15. storepb "github.com/usememos/memos/proto/gen/store"
  16. "github.com/usememos/memos/store"
  17. )
  18. type TelegramHandler struct {
  19. store *store.Store
  20. }
  21. func NewTelegramHandler(store *store.Store) *TelegramHandler {
  22. return &TelegramHandler{store: store}
  23. }
  24. func (t *TelegramHandler) BotToken(ctx context.Context) string {
  25. if setting, err := t.store.GetWorkspaceSetting(ctx, &store.FindWorkspaceSetting{
  26. Name: apiv1.SystemSettingTelegramBotTokenName.String(),
  27. }); err == nil && setting != nil {
  28. return setting.Value
  29. }
  30. return ""
  31. }
  32. const (
  33. workingMessage = "Working on sending your memo..."
  34. successMessage = "Success"
  35. )
  36. func (t *TelegramHandler) MessageHandle(ctx context.Context, bot *telegram.Bot, message telegram.Message, attachments []telegram.Attachment) error {
  37. reply, err := bot.SendReplyMessage(ctx, message.Chat.ID, message.MessageID, workingMessage)
  38. if err != nil {
  39. return errors.Wrap(err, "Failed to SendReplyMessage")
  40. }
  41. var creatorID int32
  42. userSettingList, err := t.store.ListUserSettings(ctx, &store.FindUserSetting{
  43. Key: storepb.UserSettingKey_USER_SETTING_TELEGRAM_USER_ID,
  44. })
  45. if err != nil {
  46. return errors.Wrap(err, "Failed to find userSettingList")
  47. }
  48. for _, userSetting := range userSettingList {
  49. if userSetting.GetTelegramUserId() == strconv.FormatInt(message.From.ID, 10) {
  50. creatorID = userSetting.UserId
  51. }
  52. }
  53. if creatorID == 0 {
  54. _, 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)
  55. return err
  56. }
  57. create := &store.Memo{
  58. ResourceName: shortuuid.New(),
  59. CreatorID: creatorID,
  60. Visibility: store.Private,
  61. }
  62. if message.Text != nil {
  63. create.Content = convertToMarkdown(*message.Text, message.Entities)
  64. }
  65. if message.Caption != nil {
  66. create.Content = convertToMarkdown(*message.Caption, message.CaptionEntities)
  67. }
  68. if message.ForwardFromChat != nil {
  69. create.Content += fmt.Sprintf("\n\n[Message link](%s)", message.GetMessageLink())
  70. }
  71. memoMessage, err := t.store.CreateMemo(ctx, create)
  72. if err != nil {
  73. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to CreateMemo: %s", err), nil)
  74. return err
  75. }
  76. // create resources
  77. for _, attachment := range attachments {
  78. // Fill the common field of create
  79. create := store.Resource{
  80. ResourceName: shortuuid.New(),
  81. CreatorID: creatorID,
  82. Filename: filepath.Base(attachment.FileName),
  83. Type: attachment.GetMimeType(),
  84. Size: attachment.FileSize,
  85. MemoID: &memoMessage.ID,
  86. }
  87. err := apiv1.SaveResourceBlob(ctx, t.store, &create, bytes.NewReader(attachment.Data))
  88. if err != nil {
  89. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to SaveResourceBlob: %s", err), nil)
  90. return err
  91. }
  92. _, err = t.store.CreateResource(ctx, &create)
  93. if err != nil {
  94. _, err := bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Failed to CreateResource: %s", err), nil)
  95. return err
  96. }
  97. }
  98. keyboard := generateKeyboardForMemoID(memoMessage.ID)
  99. _, err = bot.EditMessage(ctx, message.Chat.ID, reply.MessageID, fmt.Sprintf("Saved as %s Memo %d", memoMessage.Visibility, memoMessage.ID), keyboard)
  100. _ = t.dispatchMemoRelatedWebhook(ctx, *memoMessage, "memos.memo.created")
  101. return err
  102. }
  103. func (t *TelegramHandler) CallbackQueryHandle(ctx context.Context, bot *telegram.Bot, callbackQuery telegram.CallbackQuery) error {
  104. var memoID int32
  105. var visibility store.Visibility
  106. n, err := fmt.Sscanf(callbackQuery.Data, "%s %d", &visibility, &memoID)
  107. if err != nil || n != 2 {
  108. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to parse callbackQuery.Data %s", callbackQuery.Data))
  109. }
  110. update := store.UpdateMemo{
  111. ID: memoID,
  112. Visibility: &visibility,
  113. }
  114. err = t.store.UpdateMemo(ctx, &update)
  115. if err != nil {
  116. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to call UpdateMemo %s", err))
  117. }
  118. keyboard := generateKeyboardForMemoID(memoID)
  119. _, err = bot.EditMessage(ctx, callbackQuery.Message.Chat.ID, callbackQuery.Message.MessageID, fmt.Sprintf("Saved as %s Memo %d", visibility, memoID), keyboard)
  120. if err != nil {
  121. return bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Failed to EditMessage %s", err))
  122. }
  123. err = bot.AnswerCallbackQuery(ctx, callbackQuery.ID, fmt.Sprintf("Success changing Memo %d to %s", memoID, visibility))
  124. memo, webhookErr := t.store.GetMemo(ctx, &store.FindMemo{
  125. ID: &memoID,
  126. })
  127. if webhookErr == nil {
  128. _ = t.dispatchMemoRelatedWebhook(ctx, *memo, "memos.memo.updated")
  129. }
  130. return err
  131. }
  132. func generateKeyboardForMemoID(id int32) [][]telegram.InlineKeyboardButton {
  133. allVisibility := []store.Visibility{
  134. store.Public,
  135. store.Protected,
  136. store.Private,
  137. }
  138. buttons := make([]telegram.InlineKeyboardButton, 0, len(allVisibility))
  139. for _, v := range allVisibility {
  140. button := telegram.InlineKeyboardButton{
  141. Text: v.String(),
  142. CallbackData: fmt.Sprintf("%s %d", v, id),
  143. }
  144. buttons = append(buttons, button)
  145. }
  146. return [][]telegram.InlineKeyboardButton{buttons}
  147. }
  148. func convertToMarkdown(text string, messageEntities []telegram.MessageEntity) string {
  149. insertions := make(map[int]string)
  150. for _, e := range messageEntities {
  151. var before, after string
  152. // this is supported by the current markdown
  153. switch e.Type {
  154. case telegram.Bold:
  155. before = "**"
  156. after = "**"
  157. case telegram.Italic:
  158. before = "*"
  159. after = "*"
  160. case telegram.Strikethrough:
  161. before = "~~"
  162. after = "~~"
  163. case telegram.Code:
  164. before = "`"
  165. after = "`"
  166. case telegram.Pre:
  167. before = "```" + e.Language
  168. after = "```"
  169. case telegram.TextLink:
  170. before = "["
  171. after = fmt.Sprintf(`](%s)`, e.URL)
  172. }
  173. if before != "" {
  174. insertions[e.Offset] += before
  175. insertions[e.Offset+e.Length] = after + insertions[e.Offset+e.Length]
  176. }
  177. }
  178. input := []rune(text)
  179. var output []rune
  180. utf16pos := 0
  181. for i := 0; i < len(input); i++ {
  182. output = append(output, []rune(insertions[utf16pos])...)
  183. output = append(output, input[i])
  184. utf16pos += len(utf16.Encode([]rune{input[i]}))
  185. }
  186. output = append(output, []rune(insertions[utf16pos])...)
  187. return string(output)
  188. }
  189. func (t *TelegramHandler) dispatchMemoRelatedWebhook(ctx context.Context, memo store.Memo, activityType string) error {
  190. webhooks, err := t.store.ListWebhooks(ctx, &store.FindWebhook{
  191. CreatorID: &memo.CreatorID,
  192. })
  193. if err != nil {
  194. return err
  195. }
  196. for _, hook := range webhooks {
  197. payload := t.convertMemoToWebhookPayload(ctx, memo)
  198. payload.ActivityType = activityType
  199. payload.URL = hook.Url
  200. err := webhook.Post(*payload)
  201. if err != nil {
  202. return errors.Wrap(err, "failed to post webhook")
  203. }
  204. }
  205. return nil
  206. }
  207. func (t *TelegramHandler) convertMemoToWebhookPayload(ctx context.Context, memo store.Memo) (payload *webhook.WebhookPayload) {
  208. payload = &webhook.WebhookPayload{
  209. CreatorID: memo.CreatorID,
  210. CreatedTs: time.Now().Unix(),
  211. Memo: &webhook.Memo{
  212. ID: memo.ID,
  213. CreatorID: memo.CreatorID,
  214. CreatedTs: memo.CreatedTs,
  215. UpdatedTs: memo.UpdatedTs,
  216. Content: memo.Content,
  217. Visibility: memo.Visibility.String(),
  218. Pinned: memo.Pinned,
  219. ResourceList: make([]*webhook.Resource, 0),
  220. RelationList: make([]*webhook.MemoRelation, 0),
  221. },
  222. }
  223. resourceList, err := t.store.ListResources(ctx, &store.FindResource{
  224. MemoID: &memo.ID,
  225. })
  226. if err != nil {
  227. return payload
  228. }
  229. for _, resource := range resourceList {
  230. payload.Memo.ResourceList = append(payload.Memo.ResourceList, &webhook.Resource{
  231. ID: resource.ID,
  232. CreatorID: resource.CreatorID,
  233. CreatedTs: resource.CreatedTs,
  234. UpdatedTs: resource.UpdatedTs,
  235. Filename: resource.Filename,
  236. Type: resource.Type,
  237. Size: resource.Size,
  238. InternalPath: resource.InternalPath,
  239. ExternalLink: resource.ExternalLink,
  240. })
  241. }
  242. relationList, err := t.store.ListMemoRelations(ctx, &store.FindMemoRelation{
  243. MemoID: &memo.ID,
  244. })
  245. if err != nil {
  246. return payload
  247. }
  248. for _, relation := range relationList {
  249. payload.Memo.RelationList = append(payload.Memo.RelationList, &webhook.MemoRelation{
  250. MemoID: relation.MemoID,
  251. RelatedMemoID: relation.RelatedMemoID,
  252. Type: string(relation.Type),
  253. })
  254. }
  255. return payload
  256. }