edits: bridge TG -> Matrix
Signed-off-by: Sumner Evans <sumner.evans@automattic.com>
This commit is contained in:
@@ -19,7 +19,6 @@ import (
|
|||||||
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/ids"
|
"go.mau.fi/mautrix-telegram/pkg/connector/ids"
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/media"
|
"go.mau.fi/mautrix-telegram/pkg/connector/media"
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/msgconv"
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/store"
|
"go.mau.fi/mautrix-telegram/pkg/connector/store"
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/util"
|
"go.mau.fi/mautrix-telegram/pkg/connector/util"
|
||||||
)
|
)
|
||||||
@@ -33,7 +32,6 @@ type TelegramClient struct {
|
|||||||
userLogin *bridgev2.UserLogin
|
userLogin *bridgev2.UserLogin
|
||||||
client *telegram.Client
|
client *telegram.Client
|
||||||
clientCancel context.CancelFunc
|
clientCancel context.CancelFunc
|
||||||
msgConv *msgconv.MessageConverter
|
|
||||||
|
|
||||||
reactionMessageLocks map[int]*sync.Mutex
|
reactionMessageLocks map[int]*sync.Mutex
|
||||||
|
|
||||||
@@ -141,7 +139,6 @@ func NewTelegramClient(ctx context.Context, tc *TelegramConnector, login *bridge
|
|||||||
Logger: zaplog,
|
Logger: zaplog,
|
||||||
UpdateHandler: updatesManager,
|
UpdateHandler: updatesManager,
|
||||||
})
|
})
|
||||||
client.msgConv = msgconv.NewMessageConverter(client.client, tc.Bridge.Matrix, tc.Store, tc.Config.AnimatedSticker, tc.useDirectMedia)
|
|
||||||
client.clientCancel, err = connectTelegramClient(ctx, client.client)
|
client.clientCancel, err = connectTelegramClient(ctx, client.client)
|
||||||
client.reactionMessageLocks = map[int]*sync.Mutex{}
|
client.reactionMessageLocks = map[int]*sync.Mutex{}
|
||||||
go func() {
|
go func() {
|
||||||
|
|||||||
@@ -78,15 +78,11 @@ func (tg *TelegramConnector) ValidateConfig() error {
|
|||||||
|
|
||||||
func (tg *TelegramConnector) GetDBMetaTypes() database.MetaTypes {
|
func (tg *TelegramConnector) GetDBMetaTypes() database.MetaTypes {
|
||||||
return database.MetaTypes{
|
return database.MetaTypes{
|
||||||
Ghost: func() any {
|
Ghost: func() any { return &GhostMetadata{} },
|
||||||
return &GhostMetadata{}
|
Portal: nil,
|
||||||
},
|
Message: func() any { return &MessageMetadata{} },
|
||||||
Portal: nil,
|
Reaction: nil,
|
||||||
Message: nil,
|
UserLogin: func() any { return &UserLoginMetadata{} },
|
||||||
Reaction: nil,
|
|
||||||
UserLogin: func() any {
|
|
||||||
return &UserLoginMetadata{}
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -95,6 +91,10 @@ type GhostMetadata struct {
|
|||||||
AccessHash int64 `json:"access_hash"`
|
AccessHash int64 `json:"access_hash"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type MessageMetadata struct {
|
||||||
|
ContentHash []byte `json:"content_hash,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
type UserLoginMetadata struct {
|
type UserLoginMetadata struct {
|
||||||
Phone string `json:"phone"`
|
Phone string `json:"phone"`
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
package msgconv
|
package connector
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -12,9 +11,6 @@ type GeoURI struct {
|
|||||||
Long float64
|
Long float64
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ json.Unmarshaler = (*GeoURI)(nil)
|
|
||||||
var _ json.Marshaler = (*GeoURI)(nil)
|
|
||||||
|
|
||||||
func GeoURIFromLatLong(lat, long float64) GeoURI {
|
func GeoURIFromLatLong(lat, long float64) GeoURI {
|
||||||
return GeoURI{lat, long}
|
return GeoURI{lat, long}
|
||||||
}
|
}
|
||||||
@@ -42,19 +38,3 @@ func ParseGeoURI(uri string) (g GeoURI, err error) {
|
|||||||
func (g GeoURI) URI() string {
|
func (g GeoURI) URI() string {
|
||||||
return fmt.Sprintf("geo:%f,%f", g.Lat, g.Long)
|
return fmt.Sprintf("geo:%f,%f", g.Lat, g.Long)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (g *GeoURI) UnmarshalJSON(data []byte) (err error) {
|
|
||||||
var uri string
|
|
||||||
err = json.Unmarshal(data, &uri)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
geo, err := ParseGeoURI(uri)
|
|
||||||
g.Lat = geo.Lat
|
|
||||||
g.Long = geo.Long
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *GeoURI) MarshalJSON() ([]byte, error) {
|
|
||||||
return json.Marshal(g.URI())
|
|
||||||
}
|
|
||||||
@@ -13,18 +13,20 @@ import (
|
|||||||
// The format of the media ID is as follows (each character represents a single
|
// The format of the media ID is as follows (each character represents a single
|
||||||
// byte, |'s added for clarity):
|
// byte, |'s added for clarity):
|
||||||
//
|
//
|
||||||
// v|p|cccccccc|mmmmmmmm|T
|
// v|p|cccccccc|mmmmmmmm|T|MMMMMMMM
|
||||||
//
|
//
|
||||||
// v (int8) = binary encoding format version. Should be 0.
|
// v (int8) = binary encoding format version. Should be 0.
|
||||||
// p (byte) = the peer type of the Telegram chat ID
|
// p (byte) = the peer type of the Telegram chat ID
|
||||||
// cccccccc (int64) = the Telegram chat ID (big endian)
|
// cccccccc (int64) = the Telegram chat ID (big endian)
|
||||||
// mmmmmmmm (int64) = the Telegram message ID (big endian)
|
// mmmmmmmm (int64) = the Telegram message ID (big endian)
|
||||||
// T (byte) = 0 or 1 depending on whether it's a thumbnail
|
// T (byte) = 0 or 1 depending on whether it's a thumbnail (optional)
|
||||||
|
// MMMMMMMM (int64) = the Telegram media ID (big endian) (optional)
|
||||||
type DirectMediaInfo struct {
|
type DirectMediaInfo struct {
|
||||||
PeerType PeerType
|
PeerType PeerType
|
||||||
ChatID int64
|
ChatID int64
|
||||||
MessageID int64
|
MessageID int64
|
||||||
Thumbnail bool
|
Thumbnail bool
|
||||||
|
TelegramMediaID int64
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m DirectMediaInfo) AsMediaID() (networkid.MediaID, error) {
|
func (m DirectMediaInfo) AsMediaID() (networkid.MediaID, error) {
|
||||||
@@ -39,6 +41,7 @@ func (m DirectMediaInfo) AsMediaID() (networkid.MediaID, error) {
|
|||||||
} else {
|
} else {
|
||||||
mediaID = append(mediaID, 0x00)
|
mediaID = append(mediaID, 0x00)
|
||||||
}
|
}
|
||||||
|
mediaID = binary.BigEndian.AppendUint64(mediaID, uint64(m.TelegramMediaID)) // Telegram Message ID
|
||||||
return mediaID, nil
|
return mediaID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -51,7 +54,10 @@ func ParseDirectMediaInfo(mediaID networkid.MediaID) (info DirectMediaInfo, err
|
|||||||
err = fmt.Errorf("invalid version %d", mediaID[0])
|
err = fmt.Errorf("invalid version %d", mediaID[0])
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if len(mediaID) != 18 && len(mediaID) != 19 {
|
|
||||||
|
// For compatibility with old media IDs that don't have the thumbnail flag
|
||||||
|
// and the Telegram media ID, we allow media IDs with 18, 19, or 27 bytes.
|
||||||
|
if len(mediaID) != 18 && len(mediaID) != 19 && len(mediaID) != 27 {
|
||||||
err = fmt.Errorf("invalid media ID")
|
err = fmt.Errorf("invalid media ID")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -61,8 +67,11 @@ func ParseDirectMediaInfo(mediaID networkid.MediaID) (info DirectMediaInfo, err
|
|||||||
}
|
}
|
||||||
info.ChatID = int64(binary.BigEndian.Uint64(mediaID[2:]))
|
info.ChatID = int64(binary.BigEndian.Uint64(mediaID[2:]))
|
||||||
info.MessageID = int64(binary.BigEndian.Uint64(mediaID[10:]))
|
info.MessageID = int64(binary.BigEndian.Uint64(mediaID[10:]))
|
||||||
if len(mediaID) == 19 {
|
if len(mediaID) >= 19 {
|
||||||
info.Thumbnail = mediaID[18] == 1
|
info.Thumbnail = mediaID[18] == 1
|
||||||
}
|
}
|
||||||
|
if len(mediaID) >= 20 {
|
||||||
|
info.TelegramMediaID = int64(binary.BigEndian.Uint64(mediaID[19:]))
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
+12
-5
@@ -2,6 +2,7 @@ package connector
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/sha256"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -21,7 +22,6 @@ import (
|
|||||||
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/emojis"
|
"go.mau.fi/mautrix-telegram/pkg/connector/emojis"
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/ids"
|
"go.mau.fi/mautrix-telegram/pkg/connector/ids"
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/msgconv"
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/waveform"
|
"go.mau.fi/mautrix-telegram/pkg/connector/waveform"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -105,8 +105,8 @@ func (t *TelegramClient) HandleMatrixMessage(ctx context.Context, msg *bridgev2.
|
|||||||
updates, err = builder.Media(ctx, media)
|
updates, err = builder.Media(ctx, media)
|
||||||
}
|
}
|
||||||
case event.MsgLocation:
|
case event.MsgLocation:
|
||||||
var uri msgconv.GeoURI
|
var uri GeoURI
|
||||||
uri, err = msgconv.ParseGeoURI(msg.Content.GeoURI)
|
uri, err = ParseGeoURI(msg.Content.GeoURI)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -131,6 +131,9 @@ func (t *TelegramClient) HandleMatrixMessage(ctx context.Context, msg *bridgev2.
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hasher := sha256.New()
|
||||||
|
hasher.Write([]byte(msg.Content.Body))
|
||||||
|
|
||||||
var tgMessageID, tgDate int
|
var tgMessageID, tgDate int
|
||||||
switch sentMessage := updates.(type) {
|
switch sentMessage := updates.(type) {
|
||||||
case *tg.UpdateShortSentMessage:
|
case *tg.UpdateShortSentMessage:
|
||||||
@@ -139,9 +142,12 @@ func (t *TelegramClient) HandleMatrixMessage(ctx context.Context, msg *bridgev2.
|
|||||||
case *tg.Updates:
|
case *tg.Updates:
|
||||||
tgDate = sentMessage.Date
|
tgDate = sentMessage.Date
|
||||||
for _, u := range sentMessage.Updates {
|
for _, u := range sentMessage.Updates {
|
||||||
if update, ok := u.(*tg.UpdateMessageID); ok {
|
switch update := u.(type) {
|
||||||
|
case *tg.UpdateMessageID:
|
||||||
tgMessageID = update.ID
|
tgMessageID = update.ID
|
||||||
break
|
case *tg.UpdateNewMessage:
|
||||||
|
msg := update.Message.(*tg.Message)
|
||||||
|
hasher.Write(mediaHashID(msg.Media))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if tgMessageID == 0 {
|
if tgMessageID == 0 {
|
||||||
@@ -158,6 +164,7 @@ func (t *TelegramClient) HandleMatrixMessage(ctx context.Context, msg *bridgev2.
|
|||||||
Room: networkid.PortalKey{ID: msg.Portal.ID},
|
Room: networkid.PortalKey{ID: msg.Portal.ID},
|
||||||
SenderID: t.userID,
|
SenderID: t.userID,
|
||||||
Timestamp: time.Unix(int64(tgDate), 0),
|
Timestamp: time.Unix(int64(tgDate), 0),
|
||||||
|
Metadata: &MessageMetadata{ContentHash: hasher.Sum(nil)},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -326,16 +326,17 @@ func (t *ReadyTransferer) Download(ctx context.Context) ([]byte, *event.FileInfo
|
|||||||
}
|
}
|
||||||
|
|
||||||
// DirectDownloadURL returns the direct download URL for the media.
|
// DirectDownloadURL returns the direct download URL for the media.
|
||||||
func (t *ReadyTransferer) DirectDownloadURL(ctx context.Context, portal *bridgev2.Portal, msgID int, thumbnail bool) (id.ContentURIString, *event.FileInfo, error) {
|
func (t *ReadyTransferer) DirectDownloadURL(ctx context.Context, portal *bridgev2.Portal, msgID int, thumbnail bool, telegramMediaID int64) (id.ContentURIString, *event.FileInfo, error) {
|
||||||
peerType, chatID, err := ids.ParsePortalID(portal.ID)
|
peerType, chatID, err := ids.ParsePortalID(portal.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", nil, err
|
return "", nil, err
|
||||||
}
|
}
|
||||||
mediaID, err := ids.DirectMediaInfo{
|
mediaID, err := ids.DirectMediaInfo{
|
||||||
PeerType: peerType,
|
PeerType: peerType,
|
||||||
ChatID: chatID,
|
ChatID: chatID,
|
||||||
MessageID: int64(msgID),
|
MessageID: int64(msgID),
|
||||||
Thumbnail: thumbnail,
|
Thumbnail: thumbnail,
|
||||||
|
TelegramMediaID: telegramMediaID,
|
||||||
}.AsMediaID()
|
}.AsMediaID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", nil, err
|
return "", nil, err
|
||||||
|
|||||||
@@ -1,28 +0,0 @@
|
|||||||
package msgconv
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/gotd/td/telegram"
|
|
||||||
"maunium.net/go/mautrix/bridgev2"
|
|
||||||
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/media"
|
|
||||||
"go.mau.fi/mautrix-telegram/pkg/connector/store"
|
|
||||||
)
|
|
||||||
|
|
||||||
type MessageConverter struct {
|
|
||||||
client *telegram.Client
|
|
||||||
connector bridgev2.MatrixConnector
|
|
||||||
store *store.Container
|
|
||||||
animatedStickerConfig media.AnimatedStickerConfig
|
|
||||||
|
|
||||||
useDirectMedia bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewMessageConverter(client *telegram.Client, connector bridgev2.MatrixConnector, store *store.Container, animatedStickerConfig media.AnimatedStickerConfig, useDirectMedia bool) *MessageConverter {
|
|
||||||
return &MessageConverter{
|
|
||||||
client: client,
|
|
||||||
connector: connector,
|
|
||||||
store: store,
|
|
||||||
animatedStickerConfig: animatedStickerConfig,
|
|
||||||
useDirectMedia: useDirectMedia,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+42
-12
@@ -1,6 +1,7 @@
|
|||||||
package connector
|
package connector
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"slices"
|
"slices"
|
||||||
@@ -58,7 +59,7 @@ func (t *TelegramClient) onUpdateNewMessage(ctx context.Context, update IGetMess
|
|||||||
PortalKey: ids.MakePortalKey(msg.PeerID),
|
PortalKey: ids.MakePortalKey(msg.PeerID),
|
||||||
Data: msg,
|
Data: msg,
|
||||||
CreatePortal: true,
|
CreatePortal: true,
|
||||||
ConvertMessageFunc: t.msgConv.ToMatrix,
|
ConvertMessageFunc: t.convertToMatrix,
|
||||||
Timestamp: time.Unix(int64(msg.Date), 0),
|
Timestamp: time.Unix(int64(msg.Date), 0),
|
||||||
})
|
})
|
||||||
case *tg.MessageService:
|
case *tg.MessageService:
|
||||||
@@ -209,30 +210,59 @@ func (t *TelegramClient) onEntityUpdate(ctx context.Context, e tg.Entities) erro
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (t *TelegramClient) onMessageEdit(ctx context.Context, update IGetMessage) error {
|
func (t *TelegramClient) onMessageEdit(ctx context.Context, update IGetMessage) error {
|
||||||
// fmt.Printf("message edit %+v\n", update)
|
|
||||||
msg, ok := update.GetMessage().(*tg.Message)
|
msg, ok := update.GetMessage().(*tg.Message)
|
||||||
if !ok {
|
if !ok {
|
||||||
return fmt.Errorf("edit message is not *tg.Message")
|
return fmt.Errorf("edit message is not *tg.Message")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO do this async
|
||||||
if err := t.handleTelegramReactions(ctx, msg); err != nil {
|
if err := t.handleTelegramReactions(ctx, msg); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// t.main.Bridge.QueueRemoteEvent(t.userLogin, &bridgev2.SimpleRemoteEvent[*tg.Message]{
|
sender := t.getEventSender(msg)
|
||||||
// Type: bridgev2.RemoteEventEdit,
|
|
||||||
// LogContext: func(c zerolog.Context) zerolog.Context {
|
t.main.Bridge.QueueRemoteEvent(t.userLogin, &bridgev2.SimpleRemoteEvent[*tg.Message]{
|
||||||
// return c.
|
Type: bridgev2.RemoteEventEdit,
|
||||||
// Str("action", "edit message").
|
LogContext: func(c zerolog.Context) zerolog.Context {
|
||||||
// Int("message_id", message.ID)
|
return c.
|
||||||
// },
|
Str("action", "edit_message").
|
||||||
// Sender: sender,
|
Str("conversion_direction", "to_matrix").
|
||||||
// PortalKey: ids.MakePortalID(message.PeerID),
|
Int("message_id", msg.ID)
|
||||||
// })
|
},
|
||||||
|
ID: ids.MakeMessageID(msg.ID),
|
||||||
|
Sender: sender,
|
||||||
|
PortalKey: ids.MakePortalKey(msg.PeerID),
|
||||||
|
TargetMessage: ids.MakeMessageID(msg.ID),
|
||||||
|
Data: msg,
|
||||||
|
ConvertEditFunc: t.convertEdit,
|
||||||
|
Timestamp: time.Unix(int64(msg.EditDate), 0),
|
||||||
|
})
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (t *TelegramClient) convertEdit(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, existing []*database.Message, msg *tg.Message) (*bridgev2.ConvertedEdit, error) {
|
||||||
|
converted, err := t.convertToMatrix(ctx, portal, intent, msg)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(existing) != len(converted.Parts) {
|
||||||
|
return nil, fmt.Errorf("parts were added or removed in edit")
|
||||||
|
}
|
||||||
|
|
||||||
|
var ce bridgev2.ConvertedEdit
|
||||||
|
for i, part := range converted.Parts {
|
||||||
|
if bytes.Equal(existing[i].Metadata.(*MessageMetadata).ContentHash, part.DBMetadata.(*MessageMetadata).ContentHash) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
ce.ModifiedParts = append(ce.ModifiedParts, part.ToEditPart(existing[i]))
|
||||||
|
}
|
||||||
|
return &ce, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (t *TelegramClient) handleTelegramReactions(ctx context.Context, msg *tg.Message) error {
|
func (t *TelegramClient) handleTelegramReactions(ctx context.Context, msg *tg.Message) error {
|
||||||
if _, set := msg.GetReactions(); !set {
|
if _, set := msg.GetReactions(); !set {
|
||||||
// fmt.Printf("no reactions\n")
|
// fmt.Printf("no reactions\n")
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
package msgconv
|
package connector
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/sha256"
|
||||||
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
"html"
|
"html"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -30,15 +32,81 @@ type ttlable interface {
|
|||||||
GetTTLSeconds() (value int, ok bool)
|
GetTTLSeconds() (value int, ok bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) ToMatrix(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, msg *tg.Message) (*bridgev2.ConvertedMessage, error) {
|
func mediaHashID(media tg.MessageMediaClass) []byte {
|
||||||
|
switch media := media.(type) {
|
||||||
|
case *tg.MessageMediaPhoto:
|
||||||
|
return binary.BigEndian.AppendUint64(nil, uint64(media.Photo.GetID()))
|
||||||
|
case *tg.MessageMediaDocument:
|
||||||
|
return binary.BigEndian.AppendUint64(nil, uint64(media.Document.GetID()))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TelegramClient) mediaToMatrix(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, msg *tg.Message) (*bridgev2.ConvertedMessagePart, *database.DisappearingSetting, []byte, error) {
|
||||||
|
media, ok := msg.GetMedia()
|
||||||
|
if !ok {
|
||||||
|
return nil, nil, nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
switch media.TypeID() {
|
||||||
|
case tg.MessageMediaWebPageTypeID:
|
||||||
|
// Already handled in the message handling
|
||||||
|
return nil, nil, nil, nil
|
||||||
|
case tg.MessageMediaUnsupportedTypeID:
|
||||||
|
return &bridgev2.ConvertedMessagePart{
|
||||||
|
ID: networkid.PartID("unsupported_media"),
|
||||||
|
Type: event.EventMessage,
|
||||||
|
Content: &event.MessageEventContent{
|
||||||
|
MsgType: event.MsgNotice,
|
||||||
|
Body: "This message is not supported on your version of Mautrix-Telegram. Please check https://github.com/mautrix/telegram or ask your bridge administrator about possible updates.",
|
||||||
|
},
|
||||||
|
Extra: map[string]any{
|
||||||
|
"fi.mau.telegram.unsupported": true,
|
||||||
|
},
|
||||||
|
}, nil, nil, nil
|
||||||
|
case tg.MessageMediaPhotoTypeID, tg.MessageMediaDocumentTypeID:
|
||||||
|
converted, disappearingSetting, err := c.convertMediaRequiringUpload(ctx, portal, intent, msg.ID, media)
|
||||||
|
return converted, disappearingSetting, mediaHashID(media), err
|
||||||
|
case tg.MessageMediaContactTypeID:
|
||||||
|
return c.convertContact(media), nil, nil, nil
|
||||||
|
case tg.MessageMediaGeoTypeID, tg.MessageMediaGeoLiveTypeID, tg.MessageMediaVenueTypeID:
|
||||||
|
location, err := convertLocation(media)
|
||||||
|
return location, nil, nil, err
|
||||||
|
case tg.MessageMediaPollTypeID:
|
||||||
|
return convertPoll(media), nil, nil, nil
|
||||||
|
case tg.MessageMediaDiceTypeID:
|
||||||
|
return convertDice(media), nil, nil, nil
|
||||||
|
case tg.MessageMediaGameTypeID:
|
||||||
|
return convertGame(media), nil, nil, nil
|
||||||
|
case tg.MessageMediaStoryTypeID, tg.MessageMediaInvoiceTypeID, tg.MessageMediaGiveawayTypeID, tg.MessageMediaGiveawayResultsTypeID:
|
||||||
|
// TODO: support these properly
|
||||||
|
return &bridgev2.ConvertedMessagePart{
|
||||||
|
ID: networkid.PartID("story"),
|
||||||
|
Type: event.EventMessage,
|
||||||
|
Content: &event.MessageEventContent{
|
||||||
|
MsgType: event.MsgNotice,
|
||||||
|
Body: fmt.Sprintf("%s are not yet supported. Open Telegram to view.", media.TypeName()),
|
||||||
|
},
|
||||||
|
Extra: map[string]any{
|
||||||
|
"fi.mau.telegram.unsupported": true,
|
||||||
|
"fi.mau.telegram.type_id": media.TypeID(),
|
||||||
|
},
|
||||||
|
}, nil, nil, nil
|
||||||
|
default:
|
||||||
|
return nil, nil, nil, fmt.Errorf("unsupported media type %T", media)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TelegramClient) convertToMatrix(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, msg *tg.Message) (*bridgev2.ConvertedMessage, error) {
|
||||||
log := zerolog.Ctx(ctx).With().Str("conversion_direction", "to_matrix").Logger()
|
log := zerolog.Ctx(ctx).With().Str("conversion_direction", "to_matrix").Logger()
|
||||||
ctx = log.WithContext(ctx)
|
ctx = log.WithContext(ctx)
|
||||||
|
|
||||||
cm := &bridgev2.ConvertedMessage{}
|
cm := &bridgev2.ConvertedMessage{}
|
||||||
|
hasher := sha256.New()
|
||||||
if len(msg.Message) > 0 {
|
if len(msg.Message) > 0 {
|
||||||
var linkPreviews []*event.BeeperLinkPreview
|
var linkPreviews []*event.BeeperLinkPreview
|
||||||
if media, ok := msg.GetMedia(); ok && media.TypeID() == tg.MessageMediaWebPageTypeID {
|
if media, ok := msg.GetMedia(); ok && media.TypeID() == tg.MessageMediaWebPageTypeID {
|
||||||
preview, err := mc.webpageToBeeperLinkPreview(ctx, intent, media)
|
preview, err := c.webpageToBeeperLinkPreview(ctx, intent, media)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
} else if preview != nil {
|
} else if preview != nil {
|
||||||
@@ -46,81 +114,40 @@ func (mc *MessageConverter) ToMatrix(ctx context.Context, portal *bridgev2.Porta
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hasher.Write([]byte(msg.Message))
|
||||||
|
|
||||||
// TODO formatting
|
// TODO formatting
|
||||||
// TODO combine with other media
|
cm.Parts = []*bridgev2.ConvertedMessagePart{
|
||||||
cm.Parts = append(cm.Parts, &bridgev2.ConvertedMessagePart{
|
{
|
||||||
ID: networkid.PartID("caption"),
|
ID: networkid.PartID("caption"),
|
||||||
Type: event.EventMessage,
|
Type: event.EventMessage,
|
||||||
Content: &event.MessageEventContent{
|
Content: &event.MessageEventContent{
|
||||||
MsgType: event.MsgText,
|
MsgType: event.MsgText,
|
||||||
Body: msg.Message,
|
Body: msg.Message,
|
||||||
BeeperLinkPreviews: linkPreviews,
|
BeeperLinkPreviews: linkPreviews,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
if media, ok := msg.GetMedia(); ok {
|
|
||||||
switch media.TypeID() {
|
|
||||||
case tg.MessageMediaWebPageTypeID:
|
|
||||||
// Already handled above
|
|
||||||
case tg.MessageMediaUnsupportedTypeID:
|
|
||||||
cm.Parts = append(cm.Parts, &bridgev2.ConvertedMessagePart{
|
|
||||||
ID: networkid.PartID("unsupported_media"),
|
|
||||||
Type: event.EventMessage,
|
|
||||||
Content: &event.MessageEventContent{
|
|
||||||
MsgType: event.MsgNotice,
|
|
||||||
Body: "This message is not supported on your version of Mautrix-Telegram. Please check https://github.com/mautrix/telegram or ask your bridge administrator about possible updates.",
|
|
||||||
},
|
|
||||||
Extra: map[string]any{
|
|
||||||
"fi.mau.telegram.unsupported": true,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
case tg.MessageMediaPhotoTypeID, tg.MessageMediaDocumentTypeID:
|
|
||||||
mediaPart, disappearingSetting, err := mc.convertMediaRequiringUpload(ctx, portal, intent, msg.ID, media)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if disappearingSetting != nil {
|
|
||||||
cm.Disappear = *disappearingSetting
|
|
||||||
}
|
|
||||||
cm.Parts = append(cm.Parts, mediaPart)
|
|
||||||
case tg.MessageMediaContactTypeID:
|
|
||||||
cm.Parts = append(cm.Parts, mc.convertContact(media))
|
|
||||||
case tg.MessageMediaGeoTypeID, tg.MessageMediaGeoLiveTypeID, tg.MessageMediaVenueTypeID:
|
|
||||||
location, err := mc.convertLocation(media)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
cm.Parts = append(cm.Parts, location)
|
|
||||||
case tg.MessageMediaPollTypeID:
|
|
||||||
cm.Parts = append(cm.Parts, mc.convertPoll(media))
|
|
||||||
case tg.MessageMediaDiceTypeID:
|
|
||||||
cm.Parts = append(cm.Parts, mc.convertDice(media))
|
|
||||||
case tg.MessageMediaGameTypeID:
|
|
||||||
cm.Parts = append(cm.Parts, mc.convertGame(media))
|
|
||||||
|
|
||||||
case tg.MessageMediaStoryTypeID, tg.MessageMediaInvoiceTypeID, tg.MessageMediaGiveawayTypeID, tg.MessageMediaGiveawayResultsTypeID:
|
|
||||||
// TODO: support these properly
|
|
||||||
cm.Parts = append(cm.Parts, &bridgev2.ConvertedMessagePart{
|
|
||||||
ID: networkid.PartID("story"),
|
|
||||||
Type: event.EventMessage,
|
|
||||||
Content: &event.MessageEventContent{
|
|
||||||
MsgType: event.MsgNotice,
|
|
||||||
Body: fmt.Sprintf("%s are not yet supported. Open Telegram to view.", media.TypeName()),
|
|
||||||
},
|
|
||||||
Extra: map[string]any{
|
|
||||||
"fi.mau.telegram.unsupported": true,
|
|
||||||
"fi.mau.telegram.type_id": media.TypeID(),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
default:
|
|
||||||
return nil, fmt.Errorf("unsupported media type %T", media)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mediaPart, disappearingSetting, mediaHashID, err := c.mediaToMatrix(ctx, portal, intent, msg)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else if mediaPart != nil {
|
||||||
|
hasher.Write(mediaHashID)
|
||||||
|
cm.Parts = append(cm.Parts, mediaPart)
|
||||||
|
cm.MergeCaption()
|
||||||
|
|
||||||
|
if disappearingSetting != nil {
|
||||||
|
cm.Disappear = *disappearingSetting
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cm.Parts[0].DBMetadata = &MessageMetadata{ContentHash: hasher.Sum(nil)}
|
||||||
|
|
||||||
return cm, nil
|
return cm, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) webpageToBeeperLinkPreview(ctx context.Context, intent bridgev2.MatrixAPI, msgMedia tg.MessageMediaClass) (preview *event.BeeperLinkPreview, err error) {
|
func (c *TelegramClient) webpageToBeeperLinkPreview(ctx context.Context, intent bridgev2.MatrixAPI, msgMedia tg.MessageMediaClass) (preview *event.BeeperLinkPreview, err error) {
|
||||||
webpage, ok := msgMedia.(*tg.MessageMediaWebPage).Webpage.(*tg.WebPage)
|
webpage, ok := msgMedia.(*tg.MessageMediaWebPage).Webpage.(*tg.WebPage)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
@@ -136,9 +163,9 @@ func (mc *MessageConverter) webpageToBeeperLinkPreview(ctx context.Context, inte
|
|||||||
|
|
||||||
if pc, ok := webpage.GetPhoto(); ok && pc.TypeID() == tg.PhotoTypeID {
|
if pc, ok := webpage.GetPhoto(); ok && pc.TypeID() == tg.PhotoTypeID {
|
||||||
var fileInfo *event.FileInfo
|
var fileInfo *event.FileInfo
|
||||||
preview.ImageURL, preview.ImageEncryption, fileInfo, err = media.NewTransferer(mc.client.API()).
|
preview.ImageURL, preview.ImageEncryption, fileInfo, err = media.NewTransferer(c.client.API()).
|
||||||
WithPhoto(pc).
|
WithPhoto(pc).
|
||||||
Transfer(ctx, mc.store, intent)
|
Transfer(ctx, c.main.Store, intent)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -148,7 +175,7 @@ func (mc *MessageConverter) webpageToBeeperLinkPreview(ctx context.Context, inte
|
|||||||
return preview, nil
|
return preview, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, msgID int, msgMedia tg.MessageMediaClass) (*bridgev2.ConvertedMessagePart, *database.DisappearingSetting, error) {
|
func (c *TelegramClient) convertMediaRequiringUpload(ctx context.Context, portal *bridgev2.Portal, intent bridgev2.MatrixAPI, msgID int, msgMedia tg.MessageMediaClass) (*bridgev2.ConvertedMessagePart, *database.DisappearingSetting, error) {
|
||||||
log := zerolog.Ctx(ctx).With().
|
log := zerolog.Ctx(ctx).With().
|
||||||
Str("conversion_direction", "to_matrix").
|
Str("conversion_direction", "to_matrix").
|
||||||
Str("portal_id", string(portal.ID)).
|
Str("portal_id", string(portal.ID)).
|
||||||
@@ -157,10 +184,11 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
eventType := event.EventMessage
|
eventType := event.EventMessage
|
||||||
var partID networkid.PartID
|
var partID networkid.PartID
|
||||||
var content event.MessageEventContent
|
var content event.MessageEventContent
|
||||||
|
var telegramMediaID int64
|
||||||
var isSticker, isAnimatedSticker bool
|
var isSticker, isAnimatedSticker bool
|
||||||
extra := map[string]any{}
|
extra := map[string]any{}
|
||||||
|
|
||||||
transferer := media.NewTransferer(mc.client.API()).WithRoomID(portal.MXID)
|
transferer := media.NewTransferer(c.client.API()).WithRoomID(portal.MXID)
|
||||||
var mediaTransferer *media.ReadyTransferer
|
var mediaTransferer *media.ReadyTransferer
|
||||||
|
|
||||||
// Determine the filename and some other information
|
// Determine the filename and some other information
|
||||||
@@ -169,12 +197,14 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
partID = networkid.PartID("photo")
|
partID = networkid.PartID("photo")
|
||||||
content.MsgType = event.MsgImage
|
content.MsgType = event.MsgImage
|
||||||
content.Body = "image"
|
content.Body = "image"
|
||||||
|
telegramMediaID = msgMedia.Photo.GetID()
|
||||||
mediaTransferer = transferer.WithPhoto(msgMedia.Photo)
|
mediaTransferer = transferer.WithPhoto(msgMedia.Photo)
|
||||||
case *tg.MessageMediaDocument:
|
case *tg.MessageMediaDocument:
|
||||||
document, ok := msgMedia.Document.(*tg.Document)
|
document, ok := msgMedia.Document.(*tg.Document)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, nil, fmt.Errorf("unrecognized document type %T", msgMedia.Document)
|
return nil, nil, fmt.Errorf("unrecognized document type %T", msgMedia.Document)
|
||||||
}
|
}
|
||||||
|
telegramMediaID = document.GetID()
|
||||||
|
|
||||||
partID = networkid.PartID("document")
|
partID = networkid.PartID("document")
|
||||||
content.MsgType = event.MsgFile
|
content.MsgType = event.MsgFile
|
||||||
@@ -208,7 +238,7 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
transferer = transferer.WithImageSize(a)
|
transferer = transferer.WithImageSize(a)
|
||||||
case *tg.DocumentAttributeSticker:
|
case *tg.DocumentAttributeSticker:
|
||||||
isSticker = true
|
isSticker = true
|
||||||
if mc.animatedStickerConfig.Target == "webm" {
|
if c.main.Config.AnimatedSticker.Target == "webm" {
|
||||||
content.MsgType = event.MsgVideo
|
content.MsgType = event.MsgVideo
|
||||||
} else {
|
} else {
|
||||||
eventType = event.EventSticker
|
eventType = event.EventSticker
|
||||||
@@ -238,13 +268,13 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
extraInfo["fi.mau.autoplay"] = true
|
extraInfo["fi.mau.autoplay"] = true
|
||||||
extraInfo["fi.mau.hide_controls"] = true
|
extraInfo["fi.mau.hide_controls"] = true
|
||||||
extraInfo["fi.mau.no_audio"] = true
|
extraInfo["fi.mau.no_audio"] = true
|
||||||
transferer = transferer.WithStickerConfig(mc.animatedStickerConfig)
|
transferer = transferer.WithStickerConfig(c.main.Config.AnimatedSticker)
|
||||||
case *tg.DocumentAttributeAnimated:
|
case *tg.DocumentAttributeAnimated:
|
||||||
isAnimatedSticker = true
|
isAnimatedSticker = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if isAnimatedSticker || (isSticker && mc.animatedStickerConfig.Target == "webm") {
|
if isAnimatedSticker || (isSticker && c.main.Config.AnimatedSticker.Target == "webm") {
|
||||||
if isAnimatedSticker {
|
if isAnimatedSticker {
|
||||||
extraInfo["fi.mau.telegram.gif"] = true
|
extraInfo["fi.mau.telegram.gif"] = true
|
||||||
} else {
|
} else {
|
||||||
@@ -259,17 +289,17 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
var thumbnailInfo *event.FileInfo
|
var thumbnailInfo *event.FileInfo
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
thumbnailTransferer := media.NewTransferer(mc.client.API()).
|
thumbnailTransferer := media.NewTransferer(c.client.API()).
|
||||||
WithRoomID(portal.MXID).
|
WithRoomID(portal.MXID).
|
||||||
WithDocument(document, true)
|
WithDocument(document, true)
|
||||||
if mc.useDirectMedia {
|
if c.main.useDirectMedia {
|
||||||
thumbnailURL, thumbnailInfo, err = thumbnailTransferer.DirectDownloadURL(ctx, portal, msgID, true)
|
thumbnailURL, thumbnailInfo, err = thumbnailTransferer.DirectDownloadURL(ctx, portal, msgID, true, document.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Err(err).Msg("error getting direct download URL for thumbnail")
|
log.Err(err).Msg("error getting direct download URL for thumbnail")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if thumbnailURL == "" {
|
if thumbnailURL == "" {
|
||||||
thumbnailURL, thumbnailFile, thumbnailInfo, err = thumbnailTransferer.Transfer(ctx, mc.store, intent)
|
thumbnailURL, thumbnailFile, thumbnailInfo, err = thumbnailTransferer.Transfer(ctx, c.main.Store, intent)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, fmt.Errorf("error transferring thumbnail: %w", err)
|
return nil, nil, fmt.Errorf("error transferring thumbnail: %w", err)
|
||||||
}
|
}
|
||||||
@@ -286,14 +316,14 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
}
|
}
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
if mc.useDirectMedia && (!isSticker || mc.animatedStickerConfig.Target == "disable") {
|
if c.main.useDirectMedia && (!isSticker || c.main.Config.AnimatedSticker.Target == "disable") {
|
||||||
content.URL, content.Info, err = mediaTransferer.DirectDownloadURL(ctx, portal, msgID, false)
|
content.URL, content.Info, err = mediaTransferer.DirectDownloadURL(ctx, portal, msgID, false, telegramMediaID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Err(err).Msg("error getting direct download URL for media")
|
log.Err(err).Msg("error getting direct download URL for media")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if content.URL == "" {
|
if content.URL == "" {
|
||||||
content.URL, content.File, content.Info, err = mediaTransferer.Transfer(ctx, mc.store, intent)
|
content.URL, content.File, content.Info, err = mediaTransferer.Transfer(ctx, c.main.Store, intent)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, fmt.Errorf("error transferring media: %w", err)
|
return nil, nil, fmt.Errorf("error transferring media: %w", err)
|
||||||
}
|
}
|
||||||
@@ -337,7 +367,7 @@ func (mc *MessageConverter) convertMediaRequiringUpload(ctx context.Context, por
|
|||||||
}, disappearingSetting, nil
|
}, disappearingSetting, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertContact(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
func (c *TelegramClient) convertContact(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
||||||
contact := media.(*tg.MessageMediaContact)
|
contact := media.(*tg.MessageMediaContact)
|
||||||
name := util.FormatFullName(contact.FirstName, contact.LastName)
|
name := util.FormatFullName(contact.FirstName, contact.LastName)
|
||||||
formattedPhone := fmt.Sprintf("+%s", strings.TrimPrefix(contact.PhoneNumber, "+"))
|
formattedPhone := fmt.Sprintf("+%s", strings.TrimPrefix(contact.PhoneNumber, "+"))
|
||||||
@@ -350,7 +380,7 @@ func (mc *MessageConverter) convertContact(media tg.MessageMediaClass) *bridgev2
|
|||||||
content.Format = event.FormatHTML
|
content.Format = event.FormatHTML
|
||||||
content.FormattedBody = fmt.Sprintf(
|
content.FormattedBody = fmt.Sprintf(
|
||||||
`Shared contact info for <a href="https://matrix.to/#/%s">%s</a>: %s`,
|
`Shared contact info for <a href="https://matrix.to/#/%s">%s</a>: %s`,
|
||||||
mc.connector.GhostIntent(ids.MakeUserID(contact.UserID)).GetMXID(),
|
c.main.Bridge.Matrix.GhostIntent(ids.MakeUserID(contact.UserID)).GetMXID(),
|
||||||
html.EscapeString(name),
|
html.EscapeString(name),
|
||||||
html.EscapeString(formattedPhone),
|
html.EscapeString(formattedPhone),
|
||||||
)
|
)
|
||||||
@@ -376,7 +406,7 @@ type hasGeo interface {
|
|||||||
GetGeo() tg.GeoPointClass
|
GetGeo() tg.GeoPointClass
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertLocation(media tg.MessageMediaClass) (*bridgev2.ConvertedMessagePart, error) {
|
func convertLocation(media tg.MessageMediaClass) (*bridgev2.ConvertedMessagePart, error) {
|
||||||
g, ok := media.(hasGeo)
|
g, ok := media.(hasGeo)
|
||||||
if !ok || g.GetGeo().TypeID() != tg.GeoPointTypeID {
|
if !ok || g.GetGeo().TypeID() != tg.GeoPointTypeID {
|
||||||
return nil, fmt.Errorf("location didn't have geo or geo is wrong type")
|
return nil, fmt.Errorf("location didn't have geo or geo is wrong type")
|
||||||
@@ -395,7 +425,7 @@ func (mc *MessageConverter) convertLocation(media tg.MessageMediaClass) (*bridge
|
|||||||
}
|
}
|
||||||
|
|
||||||
geo := fmt.Sprintf("%f,%f", point.Lat, point.Long)
|
geo := fmt.Sprintf("%f,%f", point.Lat, point.Long)
|
||||||
geoURI := GeoURIFromLatLong(point.Lat, point.Long)
|
geoURI := GeoURIFromLatLong(point.Lat, point.Long).URI()
|
||||||
body := fmt.Sprintf("%.4f° %s, %.4f° %s", point.Lat, latChar, point.Long, longChar)
|
body := fmt.Sprintf("%.4f° %s, %.4f° %s", point.Lat, latChar, point.Long, longChar)
|
||||||
url := fmt.Sprintf("https://maps.google.com/?q=%s", geo)
|
url := fmt.Sprintf("https://maps.google.com/?q=%s", geo)
|
||||||
|
|
||||||
@@ -412,7 +442,7 @@ func (mc *MessageConverter) convertLocation(media tg.MessageMediaClass) (*bridge
|
|||||||
}
|
}
|
||||||
|
|
||||||
extra["org.matrix.msc3488.location"] = map[string]any{
|
extra["org.matrix.msc3488.location"] = map[string]any{
|
||||||
"uri": geoURI.URI(),
|
"uri": geoURI,
|
||||||
"description": note,
|
"description": note,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -421,7 +451,7 @@ func (mc *MessageConverter) convertLocation(media tg.MessageMediaClass) (*bridge
|
|||||||
Type: event.EventMessage,
|
Type: event.EventMessage,
|
||||||
Content: &event.MessageEventContent{
|
Content: &event.MessageEventContent{
|
||||||
MsgType: event.MsgLocation,
|
MsgType: event.MsgLocation,
|
||||||
GeoURI: geoURI.URI(),
|
GeoURI: geoURI,
|
||||||
Body: fmt.Sprintf("%s: %s\n%s", note, body, url),
|
Body: fmt.Sprintf("%s: %s\n%s", note, body, url),
|
||||||
Format: event.FormatHTML,
|
Format: event.FormatHTML,
|
||||||
FormattedBody: fmt.Sprintf(`%s: <a href="%s">%s</a>`, note, url, body),
|
FormattedBody: fmt.Sprintf(`%s: <a href="%s">%s</a>`, note, url, body),
|
||||||
@@ -430,7 +460,7 @@ func (mc *MessageConverter) convertLocation(media tg.MessageMediaClass) (*bridge
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertPoll(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
func convertPoll(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
||||||
// TODO (PLAT-25224) make this richer in the future once megabridge has support for polls
|
// TODO (PLAT-25224) make this richer in the future once megabridge has support for polls
|
||||||
|
|
||||||
poll := media.(*tg.MessageMediaPoll)
|
poll := media.(*tg.MessageMediaPoll)
|
||||||
@@ -453,7 +483,7 @@ func (mc *MessageConverter) convertPoll(media tg.MessageMediaClass) *bridgev2.Co
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertDice(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
func convertDice(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
||||||
roll := media.(*tg.MessageMediaDice)
|
roll := media.(*tg.MessageMediaDice)
|
||||||
|
|
||||||
var result string
|
var result string
|
||||||
@@ -524,7 +554,7 @@ func (mc *MessageConverter) convertDice(media tg.MessageMediaClass) *bridgev2.Co
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mc *MessageConverter) convertGame(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
func convertGame(media tg.MessageMediaClass) *bridgev2.ConvertedMessagePart {
|
||||||
// TODO (PLAT-25562) provide a richer experience for the game
|
// TODO (PLAT-25562) provide a richer experience for the game
|
||||||
game := media.(*tg.MessageMediaGame)
|
game := media.(*tg.MessageMediaGame)
|
||||||
return &bridgev2.ConvertedMessagePart{
|
return &bridgev2.ConvertedMessagePart{
|
||||||
Reference in New Issue
Block a user