2019-11-29 00:51:41 +00:00
|
|
|
package telegram
|
|
|
|
|
|
|
|
import (
|
2019-12-03 21:14:32 +00:00
|
|
|
"fmt"
|
2019-12-03 22:17:36 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2019-11-30 00:41:22 +00:00
|
|
|
"strconv"
|
2019-12-01 13:13:45 +00:00
|
|
|
"strings"
|
2019-12-03 21:14:32 +00:00
|
|
|
"sync"
|
2019-11-30 00:41:22 +00:00
|
|
|
|
2020-01-13 17:22:45 +00:00
|
|
|
"dev.narayana.im/narayana/telegabber/telegram/formatter"
|
2019-11-29 11:48:27 +00:00
|
|
|
"dev.narayana.im/narayana/telegabber/xmpp/gateway"
|
|
|
|
|
2019-11-29 00:51:41 +00:00
|
|
|
log "github.com/sirupsen/logrus"
|
2022-01-17 20:45:40 +00:00
|
|
|
"github.com/zelenin/go-tdlib/client"
|
2019-11-29 00:51:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func uhOh() {
|
|
|
|
log.Fatal("Update type mismatch")
|
|
|
|
}
|
|
|
|
|
2019-12-03 21:14:32 +00:00
|
|
|
func int64SliceToStringSlice(ints []int64) []string {
|
|
|
|
strings := make([]string, len(ints))
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
|
|
|
|
for i, xi := range ints {
|
|
|
|
wg.Add(1)
|
|
|
|
go func(i int, xi int64) {
|
|
|
|
strings[i] = strconv.FormatInt(xi, 10)
|
|
|
|
wg.Done()
|
|
|
|
}(i, xi)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
return strings
|
|
|
|
}
|
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
func (c *Client) getChatMessageLock(chatID int64) *sync.Mutex {
|
|
|
|
lock, ok := c.locks.chatMessageLocks[chatID]
|
|
|
|
if !ok {
|
|
|
|
lock = &sync.Mutex{}
|
|
|
|
c.locks.chatMessageLocks[chatID] = lock
|
|
|
|
}
|
|
|
|
|
|
|
|
return lock
|
|
|
|
}
|
|
|
|
|
2019-11-29 00:51:41 +00:00
|
|
|
func (c *Client) updateHandler() {
|
|
|
|
listener := c.client.GetListener()
|
|
|
|
defer listener.Close()
|
|
|
|
|
|
|
|
for update := range listener.Updates {
|
|
|
|
if update.GetClass() == client.ClassUpdate {
|
|
|
|
switch update.GetType() {
|
|
|
|
case client.TypeUpdateUser:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateUser)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateUser(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.User)
|
2019-11-29 11:48:27 +00:00
|
|
|
case client.TypeUpdateUserStatus:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateUserStatus)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateUserStatus(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.Status)
|
2019-11-30 00:41:22 +00:00
|
|
|
case client.TypeUpdateNewChat:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateNewChat)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateNewChat(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.Chat)
|
2019-12-01 13:13:45 +00:00
|
|
|
case client.TypeUpdateNewMessage:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateNewMessage)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateNewMessage(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.Message)
|
2019-12-03 21:14:32 +00:00
|
|
|
case client.TypeUpdateMessageContent:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateMessageContent)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateMessageContent(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.NewContent)
|
2019-12-03 21:14:32 +00:00
|
|
|
case client.TypeUpdateDeleteMessages:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateDeleteMessages)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateDeleteMessages(typedUpdate)
|
2019-12-03 22:17:36 +00:00
|
|
|
case client.TypeUpdateFile:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateFile)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateFile(typedUpdate)
|
2019-12-04 16:22:22 +00:00
|
|
|
log.Debugf("%#v", typedUpdate.File)
|
2019-12-17 01:56:11 +00:00
|
|
|
case client.TypeUpdateAuthorizationState:
|
|
|
|
typedUpdate, ok := update.(*client.UpdateAuthorizationState)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
c.updateAuthorizationState(typedUpdate)
|
2019-11-29 00:51:41 +00:00
|
|
|
default:
|
|
|
|
// log only handled types
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debugf("%#v", update)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// new user discovered
|
2019-11-29 00:51:41 +00:00
|
|
|
func (c *Client) updateUser(update *client.UpdateUser) {
|
2022-01-17 20:45:40 +00:00
|
|
|
c.cache.SetUser(update.User.Id, update.User)
|
2022-01-31 14:31:05 +00:00
|
|
|
show, status := c.userStatusToText(update.User.Status, update.User.Id)
|
2022-01-17 20:45:40 +00:00
|
|
|
go c.ProcessStatusUpdate(update.User.Id, status, show)
|
2019-11-29 00:51:41 +00:00
|
|
|
}
|
2019-11-29 11:48:27 +00:00
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// user status changed
|
2019-11-29 11:48:27 +00:00
|
|
|
func (c *Client) updateUserStatus(update *client.UpdateUserStatus) {
|
2022-01-31 14:31:05 +00:00
|
|
|
show, status := c.userStatusToText(update.Status, update.UserId)
|
2022-01-17 20:45:40 +00:00
|
|
|
go c.ProcessStatusUpdate(update.UserId, status, show, gateway.SPImmed(false))
|
2019-11-30 00:41:22 +00:00
|
|
|
}
|
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// new chat discovered
|
2019-11-30 00:41:22 +00:00
|
|
|
func (c *Client) updateNewChat(update *client.UpdateNewChat) {
|
2019-12-30 05:01:56 +00:00
|
|
|
go func() {
|
|
|
|
if update.Chat != nil && update.Chat.Photo != nil && update.Chat.Photo.Small != nil {
|
2022-01-27 02:09:19 +00:00
|
|
|
_, err := c.DownloadFile(update.Chat.Photo.Small.Id, 32, true)
|
2019-12-30 05:01:56 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Failed to download the chat photo")
|
|
|
|
}
|
2019-11-30 00:41:22 +00:00
|
|
|
}
|
|
|
|
|
2022-01-17 20:45:40 +00:00
|
|
|
c.cache.SetChat(update.Chat.Id, update.Chat)
|
2019-11-30 00:41:22 +00:00
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
var isChannel = false
|
2022-02-01 02:23:46 +00:00
|
|
|
chatType := update.Chat.Type.ChatTypeType()
|
|
|
|
if chatType == client.TypeChatTypeSupergroup {
|
2019-12-30 05:01:56 +00:00
|
|
|
typeSupergroup, ok := update.Chat.Type.(*client.ChatTypeSupergroup)
|
|
|
|
if !ok {
|
|
|
|
uhOh()
|
|
|
|
}
|
|
|
|
isChannel = typeSupergroup.IsChannel
|
2019-11-30 00:41:22 +00:00
|
|
|
}
|
|
|
|
|
2022-02-01 02:23:46 +00:00
|
|
|
// don't subscribe to channel posters
|
|
|
|
if !((isChannel && update.Chat.LastReadInboxMessageId == 0) ||
|
|
|
|
// don't subscribe to chats with no conversation
|
|
|
|
// (manual adding will trigger a subscribe anyway)
|
|
|
|
(update.Chat.LastReadInboxMessageId == 0 &&
|
|
|
|
update.Chat.LastReadOutboxMessageId == 0 &&
|
|
|
|
chatType == client.TypeChatTypePrivate)) {
|
2019-12-30 05:01:56 +00:00
|
|
|
gateway.SendPresence(
|
|
|
|
c.xmpp,
|
|
|
|
c.jid,
|
2022-01-17 20:45:40 +00:00
|
|
|
gateway.SPFrom(strconv.FormatInt(update.Chat.Id, 10)),
|
2019-12-30 05:01:56 +00:00
|
|
|
gateway.SPType("subscribe"),
|
|
|
|
gateway.SPNickname(update.Chat.Title),
|
|
|
|
)
|
|
|
|
}
|
2019-11-30 00:41:22 +00:00
|
|
|
|
2022-01-17 20:45:40 +00:00
|
|
|
if update.Chat.Id < 0 {
|
|
|
|
c.ProcessStatusUpdate(update.Chat.Id, update.Chat.Title, "chat")
|
2019-12-30 05:01:56 +00:00
|
|
|
}
|
|
|
|
}()
|
2019-11-29 11:48:27 +00:00
|
|
|
}
|
2019-12-01 13:13:45 +00:00
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// message received
|
2019-12-01 13:13:45 +00:00
|
|
|
func (c *Client) updateNewMessage(update *client.UpdateNewMessage) {
|
2019-12-30 05:01:56 +00:00
|
|
|
go func() {
|
|
|
|
// guarantee sequential message delivering per chat
|
2022-01-17 20:45:40 +00:00
|
|
|
lock := c.getChatMessageLock(update.Message.ChatId)
|
2019-12-30 05:01:56 +00:00
|
|
|
lock.Lock()
|
|
|
|
defer lock.Unlock()
|
|
|
|
|
|
|
|
// ignore self outgoing messages
|
|
|
|
if update.Message.IsOutgoing &&
|
|
|
|
update.Message.SendingState != nil &&
|
|
|
|
update.Message.SendingState.MessageSendingStateType() == client.TypeMessageSendingStatePending {
|
|
|
|
return
|
|
|
|
}
|
2019-12-01 13:13:45 +00:00
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
log.WithFields(log.Fields{
|
2022-01-17 20:45:40 +00:00
|
|
|
"chat_id": update.Message.ChatId,
|
2019-12-30 05:01:56 +00:00
|
|
|
}).Warn("New message from chat")
|
2019-12-01 13:13:45 +00:00
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
text := c.messageToText(update.Message)
|
|
|
|
file, filename := c.contentToFilename(update.Message.Content)
|
2019-12-01 13:13:45 +00:00
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
// download file(s)
|
|
|
|
if file != nil && !file.Local.IsDownloadingCompleted {
|
2022-01-31 07:05:42 +00:00
|
|
|
newFile, err := c.DownloadFile(file.Id, 10, true)
|
|
|
|
if err == nil {
|
|
|
|
file = newFile
|
|
|
|
}
|
2019-12-30 05:01:56 +00:00
|
|
|
}
|
|
|
|
// OTR support (I do not know why would you need it, seriously)
|
|
|
|
if !strings.HasPrefix(text, "?OTR") {
|
|
|
|
var prefix strings.Builder
|
|
|
|
prefix.WriteString(c.messageToPrefix(update.Message, c.formatContent(file, filename)))
|
|
|
|
if text != "" {
|
|
|
|
// \n if it is groupchat and message is not empty
|
2022-01-17 20:45:40 +00:00
|
|
|
if update.Message.ChatId < 0 {
|
2019-12-30 05:01:56 +00:00
|
|
|
prefix.WriteString("\n")
|
2022-01-17 20:45:40 +00:00
|
|
|
} else if update.Message.ChatId > 0 {
|
2019-12-30 05:01:56 +00:00
|
|
|
prefix.WriteString(" | ")
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix.WriteString(text)
|
2019-12-01 13:13:45 +00:00
|
|
|
}
|
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
text = prefix.String()
|
2019-12-01 13:13:45 +00:00
|
|
|
}
|
|
|
|
|
2019-12-30 05:01:56 +00:00
|
|
|
// mark message as read
|
|
|
|
c.client.ViewMessages(&client.ViewMessagesRequest{
|
2022-01-17 20:45:40 +00:00
|
|
|
ChatId: update.Message.ChatId,
|
|
|
|
MessageIds: []int64{update.Message.Id},
|
2019-12-30 05:01:56 +00:00
|
|
|
ForceRead: true,
|
|
|
|
})
|
|
|
|
// forward message to XMPP
|
2022-01-17 20:45:40 +00:00
|
|
|
gateway.SendMessage(c.jid, strconv.FormatInt(update.Message.ChatId, 10), text, c.xmpp)
|
2019-12-30 05:01:56 +00:00
|
|
|
}()
|
2019-12-01 13:13:45 +00:00
|
|
|
}
|
2019-12-03 21:14:32 +00:00
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// message content updated
|
2019-12-03 21:14:32 +00:00
|
|
|
func (c *Client) updateMessageContent(update *client.UpdateMessageContent) {
|
2021-12-18 16:04:24 +00:00
|
|
|
markupFunction := formatter.EntityToXEP0393
|
2019-12-03 21:14:32 +00:00
|
|
|
if update.NewContent.MessageContentType() == client.TypeMessageText {
|
|
|
|
textContent := update.NewContent.(*client.MessageText)
|
2022-01-17 20:45:40 +00:00
|
|
|
text := fmt.Sprintf("✎ %v | %s", update.MessageId, formatter.Format(
|
2020-01-13 17:22:45 +00:00
|
|
|
textContent.Text.Text,
|
|
|
|
formatter.SortEntities(textContent.Text.Entities),
|
|
|
|
markupFunction,
|
|
|
|
))
|
2022-01-17 20:45:40 +00:00
|
|
|
gateway.SendMessage(c.jid, strconv.FormatInt(update.ChatId, 10), text, c.xmpp)
|
2019-12-03 21:14:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-03 22:17:36 +00:00
|
|
|
// message(s) deleted
|
2019-12-03 21:14:32 +00:00
|
|
|
func (c *Client) updateDeleteMessages(update *client.UpdateDeleteMessages) {
|
|
|
|
if update.IsPermanent {
|
2022-01-17 20:45:40 +00:00
|
|
|
text := "✗ " + strings.Join(int64SliceToStringSlice(update.MessageIds), ",")
|
|
|
|
gateway.SendMessage(c.jid, strconv.FormatInt(update.ChatId, 10), text, c.xmpp)
|
2019-12-03 21:14:32 +00:00
|
|
|
}
|
|
|
|
}
|
2019-12-03 22:17:36 +00:00
|
|
|
|
|
|
|
// file downloaded
|
|
|
|
func (c *Client) updateFile(update *client.UpdateFile) {
|
|
|
|
// not really
|
|
|
|
if !update.File.Local.IsDownloadingCompleted {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := os.Symlink(
|
|
|
|
update.File.Local.Path,
|
2022-01-31 07:05:42 +00:00
|
|
|
c.formatFilePath(c.content.Path, update.File.Remote.Id, filepath.Ext(update.File.Local.Path)),
|
2019-12-03 22:17:36 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
2019-12-30 21:51:06 +00:00
|
|
|
linkErr := err.(*os.LinkError)
|
|
|
|
if linkErr.Err.Error() == "file exists" {
|
|
|
|
log.Warn(err.Error())
|
|
|
|
} else {
|
|
|
|
log.Errorf("Error creating symlink: %v", err)
|
|
|
|
}
|
2019-12-03 22:17:36 +00:00
|
|
|
}
|
|
|
|
}
|
2019-12-17 01:56:11 +00:00
|
|
|
|
|
|
|
func (c *Client) updateAuthorizationState(update *client.UpdateAuthorizationState) {
|
|
|
|
switch update.AuthorizationState.AuthorizationStateType() {
|
|
|
|
case client.TypeAuthorizationStateClosing:
|
|
|
|
log.Warn("Closing the updates listener")
|
|
|
|
case client.TypeAuthorizationStateClosed:
|
|
|
|
log.Warn("Closed the updates listener")
|
|
|
|
c.forceClose()
|
|
|
|
}
|
|
|
|
}
|