2022-10-21 14:18:02 +00:00
// mautrix-groupme - A Matrix-GroupMe puppeting bridge.
// Copyright (C) 2022 Sumner Evans, Karmanyaah Malhotra
2018-08-12 22:00:23 +00:00
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
package main
2018-08-13 20:24:44 +00:00
import (
2020-09-24 12:25:36 +00:00
"errors"
2019-05-15 20:04:09 +00:00
"fmt"
2018-08-26 14:02:32 +00:00
"strings"
2020-09-24 12:25:36 +00:00
"time"
2018-08-26 14:02:32 +00:00
2019-01-11 19:17:31 +00:00
"maunium.net/go/maulogger/v2"
2020-05-09 11:31:06 +00:00
2020-09-24 12:25:36 +00:00
"maunium.net/go/mautrix"
2020-05-09 11:31:06 +00:00
"maunium.net/go/mautrix/appservice"
2020-05-08 19:32:22 +00:00
"maunium.net/go/mautrix/event"
2019-05-15 20:04:09 +00:00
"maunium.net/go/mautrix/format"
2020-05-08 19:32:22 +00:00
"maunium.net/go/mautrix/id"
2020-07-05 15:57:03 +00:00
2021-03-03 00:46:53 +00:00
"github.com/karmanyaahm/matrix-groupme-go/database"
2021-04-18 00:57:16 +00:00
"github.com/karmanyaahm/matrix-groupme-go/types"
2018-08-13 20:24:44 +00:00
)
2018-08-12 22:00:23 +00:00
2018-08-18 19:57:08 +00:00
type MatrixHandler struct {
bridge * Bridge
as * appservice . AppService
log maulogger . Logger
cmd * CommandHandler
}
func NewMatrixHandler ( bridge * Bridge ) * MatrixHandler {
handler := & MatrixHandler {
bridge : bridge ,
2018-08-28 21:40:54 +00:00
as : bridge . AS ,
2018-08-18 19:57:08 +00:00
log : bridge . Log . Sub ( "Matrix" ) ,
cmd : NewCommandHandler ( bridge ) ,
}
2020-05-08 19:32:22 +00:00
bridge . EventProcessor . On ( event . EventMessage , handler . HandleMessage )
2020-05-08 23:03:59 +00:00
bridge . EventProcessor . On ( event . EventEncrypted , handler . HandleEncrypted )
2020-05-08 19:32:22 +00:00
bridge . EventProcessor . On ( event . EventSticker , handler . HandleMessage )
bridge . EventProcessor . On ( event . EventRedaction , handler . HandleRedaction )
bridge . EventProcessor . On ( event . StateMember , handler . HandleMembership )
bridge . EventProcessor . On ( event . StateRoomName , handler . HandleRoomMetadata )
bridge . EventProcessor . On ( event . StateRoomAvatar , handler . HandleRoomMetadata )
bridge . EventProcessor . On ( event . StateTopic , handler . HandleRoomMetadata )
2020-05-08 23:03:59 +00:00
bridge . EventProcessor . On ( event . StateEncryption , handler . HandleEncryption )
2018-08-18 19:57:08 +00:00
return handler
}
2020-05-08 23:03:59 +00:00
func ( mx * MatrixHandler ) HandleEncryption ( evt * event . Event ) {
2020-06-17 14:50:06 +00:00
defer mx . bridge . Metrics . TrackEvent ( evt . Type ) ( )
2020-05-08 23:03:59 +00:00
if evt . Content . AsEncryption ( ) . Algorithm != id . AlgorithmMegolmV1 {
return
}
portal := mx . bridge . GetPortalByMXID ( evt . RoomID )
if portal != nil && ! portal . Encrypted {
mx . log . Debugfln ( "%s enabled encryption in %s" , evt . Sender , evt . RoomID )
portal . Encrypted = true
portal . Update ( )
}
}
2020-07-05 15:57:03 +00:00
func ( mx * MatrixHandler ) joinAndCheckMembers ( evt * event . Event , intent * appservice . IntentAPI ) * mautrix . RespJoinedMembers {
2020-05-08 19:32:22 +00:00
resp , err := intent . JoinRoomByID ( evt . RoomID )
2018-08-16 12:59:18 +00:00
if err != nil {
2020-07-05 15:57:03 +00:00
mx . log . Debugfln ( "Failed to join room %s as %s with invite from %s: %v" , evt . RoomID , intent . UserID , evt . Sender , err )
return nil
2018-08-16 12:59:18 +00:00
}
2018-08-16 16:20:07 +00:00
members , err := intent . JoinedMembers ( resp . RoomID )
2018-08-16 12:59:18 +00:00
if err != nil {
2020-07-05 15:57:03 +00:00
mx . log . Debugfln ( "Failed to get members in room %s after accepting invite from %s as %s: %v" , resp . RoomID , evt . Sender , intent . UserID , err )
2020-06-25 14:01:40 +00:00
_ , _ = intent . LeaveRoom ( resp . RoomID )
2020-07-05 15:57:03 +00:00
return nil
2018-08-16 12:59:18 +00:00
}
if len ( members . Joined ) < 2 {
2020-07-05 15:57:03 +00:00
mx . log . Debugln ( "Leaving empty room" , resp . RoomID , "after accepting invite from" , evt . Sender , "as" , intent . UserID )
2020-06-25 14:01:40 +00:00
_ , _ = intent . LeaveRoom ( resp . RoomID )
2020-07-05 15:57:03 +00:00
return nil
}
return members
}
func ( mx * MatrixHandler ) HandleBotInvite ( evt * event . Event ) {
intent := mx . as . BotIntent ( )
user := mx . bridge . GetUserByMXID ( evt . Sender )
if user == nil {
return
}
members := mx . joinAndCheckMembers ( evt , intent )
if members == nil {
2018-08-16 12:59:18 +00:00
return
}
2018-08-16 21:11:28 +00:00
2018-08-26 14:08:37 +00:00
if ! user . Whitelisted {
2020-07-05 15:57:03 +00:00
_ , _ = intent . SendNotice ( evt . RoomID , "You are not whitelisted to use this bridge.\n" +
2018-08-26 14:08:37 +00:00
"If you're the owner of this bridge, see the bridge.permissions section in your config file." )
2020-07-05 15:57:03 +00:00
_ , _ = intent . LeaveRoom ( evt . RoomID )
2018-08-26 14:08:37 +00:00
return
}
2019-11-10 19:22:11 +00:00
if evt . RoomID == mx . bridge . Config . Bridge . Relaybot . ManagementRoom {
2020-06-25 14:01:40 +00:00
_ , _ = intent . SendNotice ( evt . RoomID , "This is the relaybot management room. Send `!wa help` to get a list of commands." )
2019-11-10 19:22:11 +00:00
mx . log . Debugln ( "Joined relaybot management room" , evt . RoomID , "after invite from" , evt . Sender )
return
}
2018-08-16 21:11:28 +00:00
hasPuppets := false
2018-08-16 12:59:18 +00:00
for mxid , _ := range members . Joined {
2018-08-16 16:20:07 +00:00
if mxid == intent . UserID || mxid == evt . Sender {
2018-08-16 12:59:18 +00:00
continue
2020-05-08 19:32:22 +00:00
} else if _ , ok := mx . bridge . ParsePuppetMXID ( mxid ) ; ok {
2018-08-16 21:11:28 +00:00
hasPuppets = true
2018-08-16 12:59:18 +00:00
continue
}
2020-07-05 15:57:03 +00:00
mx . log . Debugln ( "Leaving multi-user room" , evt . RoomID , "after accepting invite from" , evt . Sender )
_ , _ = intent . SendNotice ( evt . RoomID , "This bridge is user-specific, please don't invite me into rooms with other users." )
_ , _ = intent . LeaveRoom ( evt . RoomID )
2018-08-16 12:59:18 +00:00
return
}
2020-07-10 12:23:32 +00:00
if ! hasPuppets && ( len ( user . ManagementRoom ) == 0 || evt . Content . AsMember ( ) . IsDirect ) {
2020-07-05 15:57:03 +00:00
user . SetManagementRoom ( evt . RoomID )
_ , _ = intent . SendNotice ( user . ManagementRoom , "This room has been registered as your bridge management/status room. Send `help` to get a list of commands." )
mx . log . Debugln ( evt . RoomID , "registered as a management room with" , evt . Sender )
}
}
2020-07-10 12:23:32 +00:00
func ( mx * MatrixHandler ) handlePrivatePortal ( roomID id . RoomID , inviter * User , puppet * Puppet , key database . PortalKey ) {
portal := mx . bridge . GetPortalByJID ( key )
if len ( portal . MXID ) == 0 {
mx . createPrivatePortalFromInvite ( roomID , inviter , puppet , portal )
return
}
2020-07-05 15:57:03 +00:00
err := portal . MainIntent ( ) . EnsureInvited ( portal . MXID , inviter . MXID )
if err != nil {
mx . log . Warnfln ( "Failed to invite %s to existing private chat portal %s with %s: %v. Redirecting portal to new room..." , inviter . MXID , portal . MXID , puppet . JID , err )
2020-07-10 12:23:32 +00:00
mx . createPrivatePortalFromInvite ( roomID , inviter , puppet , portal )
2020-07-05 15:57:03 +00:00
return
}
intent := puppet . DefaultIntent ( )
_ , _ = intent . SendNotice ( roomID , "You already have a private chat portal with me at %s" )
mx . log . Debugln ( "Leaving private chat room" , roomID , "as" , puppet . MXID , "after accepting invite from" , inviter . MXID , "as we already have chat with the user" )
_ , _ = intent . LeaveRoom ( roomID )
}
2020-07-10 12:23:32 +00:00
func ( mx * MatrixHandler ) createPrivatePortalFromInvite ( roomID id . RoomID , inviter * User , puppet * Puppet , portal * Portal ) {
2020-07-05 15:57:03 +00:00
portal . MXID = roomID
portal . Topic = "WhatsApp private chat"
2021-05-01 22:22:02 +00:00
portal . Key = database . PortalKey { puppet . JID , inviter . JID }
2020-07-05 15:57:03 +00:00
_ , _ = portal . MainIntent ( ) . SetRoomTopic ( portal . MXID , portal . Topic )
if portal . bridge . Config . Bridge . PrivateChatPortalMeta {
2021-04-18 00:57:16 +00:00
m , _ := mx . bridge . StateStore . TryGetMemberRaw ( portal . MXID , puppet . MXID )
portal . Name = m . DisplayName
portal . AvatarURL = types . ContentURI { id . MustParseContentURI ( m . AvatarURL ) }
2021-02-13 05:53:35 +00:00
print ( "possible bug with pointer above" )
2021-04-18 00:57:16 +00:00
portal . Avatar = m . Avatar
2020-07-05 15:57:03 +00:00
_ , _ = portal . MainIntent ( ) . SetRoomName ( portal . MXID , portal . Name )
2021-03-03 20:51:08 +00:00
_ , _ = portal . MainIntent ( ) . SetRoomAvatar ( portal . MXID , portal . AvatarURL . ContentURI )
2020-07-05 15:57:03 +00:00
} else {
portal . Name = ""
}
portal . log . Infoln ( "Created private chat portal in %s after invite from" , roomID , inviter . MXID )
intent := puppet . DefaultIntent ( )
if mx . bridge . Config . Bridge . Encryption . Default {
_ , err := intent . InviteUser ( roomID , & mautrix . ReqInviteUser { UserID : mx . bridge . Bot . UserID } )
if err != nil {
portal . log . Warnln ( "Failed to invite bridge bot to enable e2be:" , err )
}
err = mx . bridge . Bot . EnsureJoined ( roomID )
if err != nil {
portal . log . Warnln ( "Failed to join as bridge bot to enable e2be:" , err )
}
_ , err = intent . SendStateEvent ( roomID , event . StateEncryption , "" , & event . EncryptionEventContent { Algorithm : id . AlgorithmMegolmV1 } )
if err != nil {
portal . log . Warnln ( "Failed to enable e2be:" , err )
}
mx . as . StateStore . SetMembership ( roomID , inviter . MXID , event . MembershipJoin )
mx . as . StateStore . SetMembership ( roomID , puppet . MXID , event . MembershipJoin )
mx . as . StateStore . SetMembership ( roomID , mx . bridge . Bot . UserID , event . MembershipJoin )
portal . Encrypted = true
}
portal . Update ( )
portal . UpdateBridgeInfo ( )
_ , _ = intent . SendNotice ( roomID , "Private chat portal created" )
err := portal . FillInitialHistory ( inviter )
if err != nil {
portal . log . Errorln ( "Failed to fill history:" , err )
}
inviter . addPortalToCommunity ( portal )
inviter . addPuppetToCommunity ( puppet )
}
func ( mx * MatrixHandler ) HandlePuppetInvite ( evt * event . Event , inviter * User , puppet * Puppet ) {
intent := puppet . DefaultIntent ( )
members := mx . joinAndCheckMembers ( evt , intent )
if members == nil {
return
}
var hasBridgeBot , hasOtherUsers bool
for mxid , _ := range members . Joined {
if mxid == intent . UserID || mxid == inviter . MXID {
continue
} else if mxid == mx . bridge . Bot . UserID {
hasBridgeBot = true
} else {
hasOtherUsers = true
}
}
if ! hasBridgeBot && ! hasOtherUsers {
key := database . NewPortalKey ( puppet . JID , inviter . JID )
2020-07-10 12:23:32 +00:00
mx . handlePrivatePortal ( evt . RoomID , inviter , puppet , key )
2020-07-05 15:57:03 +00:00
} else if ! hasBridgeBot {
mx . log . Debugln ( "Leaving multi-user room" , evt . RoomID , "as" , puppet . MXID , "after accepting invite from" , evt . Sender )
_ , _ = intent . SendNotice ( evt . RoomID , "Please invite the bridge bot first if you want to bridge to a WhatsApp group." )
_ , _ = intent . LeaveRoom ( evt . RoomID )
} else {
_ , _ = intent . SendNotice ( evt . RoomID , "This puppet will remain inactive until this room is bridged to a WhatsApp group." )
2018-08-16 12:59:18 +00:00
}
}
2020-05-08 19:32:22 +00:00
func ( mx * MatrixHandler ) HandleMembership ( evt * event . Event ) {
2020-05-12 19:25:55 +00:00
if _ , isPuppet := mx . bridge . ParsePuppetMXID ( evt . Sender ) ; evt . Sender == mx . bridge . Bot . UserID || isPuppet {
return
}
2020-06-17 14:50:06 +00:00
defer mx . bridge . Metrics . TrackEvent ( evt . Type ) ( )
2020-05-12 19:25:55 +00:00
2020-05-08 23:03:59 +00:00
if mx . bridge . Crypto != nil {
mx . bridge . Crypto . HandleMemberEvent ( evt )
}
2020-05-08 19:32:22 +00:00
content := evt . Content . AsMember ( )
if content . Membership == event . MembershipInvite && id . UserID ( evt . GetStateKey ( ) ) == mx . as . BotMXID ( ) {
2018-08-18 19:57:08 +00:00
mx . HandleBotInvite ( evt )
2020-06-25 20:33:11 +00:00
return
2018-08-16 21:11:28 +00:00
}
2019-05-16 17:14:32 +00:00
2020-07-05 15:57:03 +00:00
user := mx . bridge . GetUserByMXID ( evt . Sender )
if user == nil || ! user . Whitelisted || ! user . IsConnected ( ) {
2019-05-16 17:14:32 +00:00
return
}
2020-07-05 15:57:03 +00:00
portal := mx . bridge . GetPortalByMXID ( evt . RoomID )
if portal == nil {
puppet := mx . bridge . GetPuppetByMXID ( id . UserID ( evt . GetStateKey ( ) ) )
if content . Membership == event . MembershipInvite && puppet != nil {
mx . HandlePuppetInvite ( evt , user , puppet )
}
2019-05-16 17:14:32 +00:00
return
}
2020-06-25 20:58:35 +00:00
isSelf := id . UserID ( evt . GetStateKey ( ) ) == evt . Sender
2020-05-08 19:32:22 +00:00
if content . Membership == event . MembershipLeave {
2020-06-25 20:58:35 +00:00
if isSelf {
2020-05-08 19:32:22 +00:00
if evt . Unsigned . PrevContent != nil {
_ = evt . Unsigned . PrevContent . ParseRaw ( evt . Type )
prevContent , ok := evt . Unsigned . PrevContent . Parsed . ( * event . MemberEventContent )
if ok {
if portal . IsPrivateChat ( ) || prevContent . Membership == "join" {
portal . HandleMatrixLeave ( user )
}
}
2019-05-16 17:14:32 +00:00
}
} else {
portal . HandleMatrixKick ( user , evt )
}
2020-06-25 20:58:35 +00:00
} else if content . Membership == event . MembershipInvite && ! isSelf {
portal . HandleMatrixInvite ( user , evt )
2019-05-16 17:14:32 +00:00
}
2018-08-16 12:59:18 +00:00
}
2020-05-08 19:32:22 +00:00
func ( mx * MatrixHandler ) HandleRoomMetadata ( evt * event . Event ) {
2021-02-13 05:53:35 +00:00
// defer mx.bridge.Metrics.TrackEvent(evt.Type)()
// user := mx.bridge.GetUserByMXID(evt.Sender)
// if user == nil || !user.Whitelisted || !user.IsConnected() {
// return
// }
// portal := mx.bridge.GetPortalByMXID(evt.RoomID)
// if portal == nil || portal.IsPrivateChat() {
// return
// }
// var resp <-chan string
// var err error
// switch content := evt.Content.Parsed.(type) {
// case *event.RoomNameEventContent:
// resp, err = user.Conn.UpdateGroupSubject(content.Name, portal.Key.JID)
// case *event.TopicEventContent:
// resp, err = user.Conn.UpdateGroupDescription(portal.Key.JID, content.Topic)
// case *event.RoomAvatarEventContent:
// return
// }
// if err != nil {
// mx.log.Errorln(err)
// } else {
// out := <-resp
// mx.log.Infoln(out)
// }
2018-08-26 14:02:32 +00:00
}
2020-05-08 23:03:59 +00:00
func ( mx * MatrixHandler ) shouldIgnoreEvent ( evt * event . Event ) bool {
2018-09-01 20:38:03 +00:00
if _ , isPuppet := mx . bridge . ParsePuppetMXID ( evt . Sender ) ; evt . Sender == mx . bridge . Bot . UserID || isPuppet {
2020-05-08 23:03:59 +00:00
return true
2018-09-01 20:38:03 +00:00
}
2019-05-23 23:33:26 +00:00
isCustomPuppet , ok := evt . Content . Raw [ "net.maunium.whatsapp.puppet" ] . ( bool )
if ok && isCustomPuppet && mx . bridge . GetPuppetByCustomMXID ( evt . Sender ) != nil {
2020-05-08 23:03:59 +00:00
return true
}
user := mx . bridge . GetUserByMXID ( evt . Sender )
if ! user . RelaybotWhitelisted {
return true
}
return false
}
2020-10-05 19:32:35 +00:00
const sessionWaitTimeout = 5 * time . Second
2020-09-24 12:25:36 +00:00
2020-05-08 23:03:59 +00:00
func ( mx * MatrixHandler ) HandleEncrypted ( evt * event . Event ) {
2021-02-21 05:58:50 +00:00
println ( "IDK iF encryption works yet" )
2020-06-17 14:50:06 +00:00
defer mx . bridge . Metrics . TrackEvent ( evt . Type ) ( )
2020-05-08 23:03:59 +00:00
if mx . shouldIgnoreEvent ( evt ) || mx . bridge . Crypto == nil {
2019-05-23 23:33:26 +00:00
return
}
2018-09-01 20:38:03 +00:00
2020-05-08 23:03:59 +00:00
decrypted , err := mx . bridge . Crypto . Decrypt ( evt )
2020-10-05 19:32:35 +00:00
if errors . Is ( err , NoSessionFound ) {
2020-09-24 12:25:36 +00:00
content := evt . Content . AsEncrypted ( )
2020-10-05 19:32:35 +00:00
mx . log . Debugfln ( "Couldn't find session %s trying to decrypt %s, waiting %d seconds..." , content . SessionID , evt . ID , int ( sessionWaitTimeout . Seconds ( ) ) )
if mx . bridge . Crypto . WaitForSession ( evt . RoomID , content . SenderKey , content . SessionID , sessionWaitTimeout ) {
mx . log . Debugfln ( "Got session %s after waiting, trying to decrypt %s again" , content . SessionID , evt . ID )
2020-09-24 12:25:36 +00:00
decrypted , err = mx . bridge . Crypto . Decrypt ( evt )
2020-10-05 19:32:35 +00:00
} else {
go mx . waitLongerForSession ( evt )
return
2020-09-24 12:25:36 +00:00
}
}
2020-05-08 23:03:59 +00:00
if err != nil {
2020-05-09 17:23:30 +00:00
mx . log . Warnfln ( "Failed to decrypt %s: %v" , evt . ID , err )
2020-09-17 19:01:17 +00:00
_ , _ = mx . bridge . Bot . SendNotice ( evt . RoomID , fmt . Sprintf (
2020-10-05 19:32:35 +00:00
"\u26a0 Your message was not bridged: %v" , err ) )
2020-05-08 23:03:59 +00:00
return
}
mx . bridge . EventProcessor . Dispatch ( decrypted )
}
2018-08-18 19:57:08 +00:00
2020-10-05 19:32:35 +00:00
func ( mx * MatrixHandler ) waitLongerForSession ( evt * event . Event ) {
const extendedTimeout = sessionWaitTimeout * 2
content := evt . Content . AsEncrypted ( )
mx . log . Debugfln ( "Couldn't find session %s trying to decrypt %s, waiting %d more seconds..." ,
content . SessionID , evt . ID , int ( extendedTimeout . Seconds ( ) ) )
resp , err := mx . bridge . Bot . SendNotice ( evt . RoomID , fmt . Sprintf (
"\u26a0 Your message was not bridged: the bridge hasn't received the decryption keys. " +
"The bridge will retry for %d seconds. If this error keeps happening, try restarting your client." ,
int ( extendedTimeout . Seconds ( ) ) ) )
if err != nil {
mx . log . Errorfln ( "Failed to send decryption error to %s: %v" , evt . RoomID , err )
}
update := event . MessageEventContent { MsgType : event . MsgNotice }
if mx . bridge . Crypto . WaitForSession ( evt . RoomID , content . SenderKey , content . SessionID , extendedTimeout ) {
mx . log . Debugfln ( "Got session %s after waiting more, trying to decrypt %s again" , content . SessionID , evt . ID )
decrypted , err := mx . bridge . Crypto . Decrypt ( evt )
2020-11-14 17:39:14 +00:00
if err == nil {
2020-10-05 19:32:35 +00:00
mx . bridge . EventProcessor . Dispatch ( decrypted )
_ , _ = mx . bridge . Bot . RedactEvent ( evt . RoomID , resp . EventID )
return
}
mx . log . Warnfln ( "Failed to decrypt %s: %v" , err )
update . Body = fmt . Sprintf ( "\u26a0 Your message was not bridged: %v" , err )
} else {
mx . log . Debugfln ( "Didn't get %s, giving up on %s" , content . SessionID , evt . ID )
update . Body = "\u26a0 Your message was not bridged: the bridge hasn't received the decryption keys. " +
"If this keeps happening, try restarting your client."
}
newContent := update
update . NewContent = & newContent
if resp != nil {
update . RelatesTo = & event . RelatesTo {
Type : event . RelReplace ,
EventID : resp . EventID ,
}
}
_ , _ = mx . bridge . Bot . SendMessageEvent ( evt . RoomID , event . EventMessage , & update )
}
2020-05-08 23:03:59 +00:00
func ( mx * MatrixHandler ) HandleMessage ( evt * event . Event ) {
2020-06-17 14:50:06 +00:00
defer mx . bridge . Metrics . TrackEvent ( evt . Type ) ( )
2020-05-08 23:03:59 +00:00
if mx . shouldIgnoreEvent ( evt ) {
2018-08-26 14:08:37 +00:00
return
}
2020-05-08 23:03:59 +00:00
user := mx . bridge . GetUserByMXID ( evt . Sender )
2020-05-08 19:32:22 +00:00
content := evt . Content . AsMessage ( )
if user . Whitelisted && content . MsgType == event . MsgText {
2018-08-18 19:57:08 +00:00
commandPrefix := mx . bridge . Config . Bridge . CommandPrefix
2020-05-08 19:32:22 +00:00
hasCommandPrefix := strings . HasPrefix ( content . Body , commandPrefix )
2018-08-18 19:57:08 +00:00
if hasCommandPrefix {
2020-05-08 19:32:22 +00:00
content . Body = strings . TrimLeft ( content . Body [ len ( commandPrefix ) : ] , " " )
2018-08-18 19:57:08 +00:00
}
2020-05-08 19:32:22 +00:00
if hasCommandPrefix || evt . RoomID == user . ManagementRoom {
mx . cmd . Handle ( evt . RoomID , user , content . Body )
2018-08-18 19:57:08 +00:00
return
}
}
2020-05-08 19:32:22 +00:00
portal := mx . bridge . GetPortalByMXID ( evt . RoomID )
2019-11-10 19:22:11 +00:00
if portal != nil && ( user . Whitelisted || portal . HasRelaybot ( ) ) {
2018-08-28 21:40:54 +00:00
portal . HandleMatrixMessage ( user , evt )
2018-08-18 19:57:08 +00:00
}
2018-08-12 22:00:23 +00:00
}
2019-05-15 22:59:36 +00:00
2020-05-08 19:32:22 +00:00
func ( mx * MatrixHandler ) HandleRedaction ( evt * event . Event ) {
2020-06-17 14:50:06 +00:00
defer mx . bridge . Metrics . TrackEvent ( evt . Type ) ( )
2019-05-15 22:59:36 +00:00
if _ , isPuppet := mx . bridge . ParsePuppetMXID ( evt . Sender ) ; evt . Sender == mx . bridge . Bot . UserID || isPuppet {
return
}
2020-05-08 23:03:59 +00:00
user := mx . bridge . GetUserByMXID ( evt . Sender )
2019-05-15 22:59:36 +00:00
if ! user . Whitelisted {
return
}
2019-08-24 19:39:12 +00:00
if ! user . HasSession ( ) {
2019-05-15 22:59:36 +00:00
return
2019-08-24 19:39:12 +00:00
} else if ! user . IsConnected ( ) {
2019-11-10 19:22:11 +00:00
msg := format . RenderMarkdown ( fmt . Sprintf ( "[%[1]s](https://matrix.to/#/%[1]s): \u26a0 " +
"You are not connected to WhatsApp, so your redaction was not bridged. " +
2020-05-08 19:32:22 +00:00
"Use `%[2]s reconnect` to reconnect." , user . MXID , mx . bridge . Config . Bridge . CommandPrefix ) , true , false )
msg . MsgType = event . MsgNotice
_ , _ = mx . bridge . Bot . SendMessageEvent ( evt . RoomID , event . EventMessage , msg )
2019-05-15 22:59:36 +00:00
return
}
2020-05-08 19:32:22 +00:00
portal := mx . bridge . GetPortalByMXID ( evt . RoomID )
2019-05-15 22:59:36 +00:00
if portal != nil {
portal . HandleMatrixRedaction ( user , evt )
}
}