0670c2b2bc
Signed-off-by: Sumner Evans <sumner.evans@automattic.com>
150 lines
4.5 KiB
Go
150 lines
4.5 KiB
Go
package connector
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
|
|
"github.com/gotd/td/tg"
|
|
"github.com/rs/zerolog"
|
|
"maunium.net/go/mautrix/bridgev2"
|
|
"maunium.net/go/mautrix/bridgev2/networkid"
|
|
"maunium.net/go/mautrix/mediaproxy"
|
|
|
|
"go.mau.fi/mautrix-telegram/pkg/connector/ids"
|
|
"go.mau.fi/mautrix-telegram/pkg/connector/media"
|
|
)
|
|
|
|
var _ bridgev2.DirectMediableNetwork = (*TelegramConnector)(nil)
|
|
|
|
type getMessages interface {
|
|
GetMessages() []tg.MessageClass
|
|
}
|
|
|
|
func (tc *TelegramConnector) Download(ctx context.Context, mediaID networkid.MediaID) (mediaproxy.GetMediaResponse, error) {
|
|
info, err := ids.ParseDirectMediaInfo(mediaID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
log := zerolog.Ctx(ctx).With().
|
|
Str("component", "direct download").
|
|
Any("info", info).
|
|
Logger()
|
|
ctx = log.WithContext(ctx)
|
|
log.Info().Msg("handling direct download")
|
|
|
|
userLogin, err := tc.Bridge.GetExistingUserLoginByID(ctx, ids.MakeUserLoginID(info.ReceiverID))
|
|
if err != nil {
|
|
if info.PeerType != ids.PeerTypeChannel {
|
|
return nil, fmt.Errorf("failed to get user login: %w", err)
|
|
}
|
|
|
|
logins, err := tc.Bridge.GetUserLoginsInPortal(ctx, ids.PeerTypeChannel.AsPortalKey(info.ChatID, ""))
|
|
if err != nil {
|
|
return nil, err
|
|
} else if len(logins) == 0 {
|
|
return nil, fmt.Errorf("no user logins in the portal (%s %d)", ids.PeerTypeChannel, info.ChatID)
|
|
}
|
|
userLogin = logins[0]
|
|
}
|
|
|
|
client := userLogin.Client.(*TelegramClient)
|
|
var messages tg.ModifiedMessagesMessages
|
|
switch info.PeerType {
|
|
case ids.PeerTypeUser, ids.PeerTypeChat:
|
|
messages, err = APICallWithUpdates(ctx, client, func() (tg.ModifiedMessagesMessages, error) {
|
|
m, err := client.client.API().MessagesGetMessages(ctx, []tg.InputMessageClass{
|
|
&tg.InputMessageID{ID: int(info.MessageID)},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
} else if messages, ok := m.(tg.ModifiedMessagesMessages); !ok {
|
|
return nil, fmt.Errorf("unsupported messages type %T", messages)
|
|
} else {
|
|
return messages, nil
|
|
}
|
|
})
|
|
case ids.PeerTypeChannel:
|
|
var accessHash int64
|
|
var found bool
|
|
accessHash, found, err = client.ScopedStore.GetChannelAccessHash(ctx, client.telegramUserID, info.ChatID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get channel access hash: %w", err)
|
|
} else if !found {
|
|
return nil, fmt.Errorf("channel access hash not found for %d", info.ChatID)
|
|
} else {
|
|
messages, err = APICallWithUpdates(ctx, client, func() (tg.ModifiedMessagesMessages, error) {
|
|
m, err := client.client.API().ChannelsGetMessages(ctx, &tg.ChannelsGetMessagesRequest{
|
|
Channel: &tg.InputChannel{ChannelID: info.ChatID, AccessHash: accessHash},
|
|
ID: []tg.InputMessageClass{
|
|
&tg.InputMessageID{ID: int(info.MessageID)},
|
|
},
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
} else if messages, ok := m.(tg.ModifiedMessagesMessages); !ok {
|
|
return nil, fmt.Errorf("unsupported messages type %T", messages)
|
|
} else {
|
|
return messages, nil
|
|
}
|
|
})
|
|
}
|
|
default:
|
|
return nil, fmt.Errorf("unknown peer type %s", info.PeerType)
|
|
}
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to get messages for %+v: %w", info, err)
|
|
}
|
|
|
|
var msgMedia tg.MessageMediaClass
|
|
var found bool
|
|
for _, message := range messages.GetMessages() {
|
|
if msg, ok := message.(*tg.Message); ok && msg.ID == int(info.MessageID) {
|
|
msgMedia = msg.Media
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
return nil, fmt.Errorf("no media found with ID %d", info.MessageID)
|
|
}
|
|
|
|
transferer := media.NewTransferer(client.client.API())
|
|
var readyTransferer *media.ReadyTransferer
|
|
switch msgMedia := msgMedia.(type) {
|
|
case *tg.MessageMediaPhoto:
|
|
readyTransferer = transferer.WithPhoto(msgMedia.Photo)
|
|
case *tg.MessageMediaDocument:
|
|
document, ok := msgMedia.Document.(*tg.Document)
|
|
if !ok {
|
|
return nil, fmt.Errorf("unknown document type %T", msgMedia.Document)
|
|
}
|
|
for _, attr := range document.GetAttributes() {
|
|
if attr.TypeID() == tg.DocumentAttributeStickerTypeID {
|
|
transferer = transferer.WithStickerConfig(tc.Config.AnimatedSticker)
|
|
}
|
|
}
|
|
|
|
readyTransferer = transferer.WithDocument(msgMedia.Document, info.Thumbnail)
|
|
default:
|
|
return nil, fmt.Errorf("unhandled media type %T", msgMedia)
|
|
}
|
|
|
|
data, fileInfo, err := readyTransferer.Download(ctx)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to download media")
|
|
return nil, err
|
|
}
|
|
|
|
return &mediaproxy.GetMediaResponseData{
|
|
Reader: io.NopCloser(bytes.NewBuffer(data)),
|
|
ContentType: fileInfo.MimeType,
|
|
ContentLength: int64(fileInfo.Size),
|
|
}, nil
|
|
}
|
|
|
|
func (tg *TelegramConnector) SetUseDirectMedia() {
|
|
tg.useDirectMedia = true
|
|
}
|