2018-08-13 20:24:44 +00:00
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
2019-01-12 13:54:04 +00:00
// Copyright (C) 2019 Tulir Asokan
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 (
2018-08-26 14:02:32 +00:00
"strings"
2019-01-11 19:17:31 +00:00
"maunium.net/go/maulogger/v2"
"maunium.net/go/mautrix"
2018-08-26 14:02:32 +00:00
"maunium.net/go/mautrix-appservice"
2019-01-11 19:17:31 +00:00
2018-08-26 14:02:32 +00:00
"maunium.net/go/mautrix-whatsapp/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 ) ,
}
2019-01-11 19:17:31 +00:00
bridge . EventProcessor . On ( mautrix . EventMessage , handler . HandleMessage )
bridge . EventProcessor . On ( mautrix . StateMember , handler . HandleMembership )
bridge . EventProcessor . On ( mautrix . StateRoomName , handler . HandleRoomMetadata )
bridge . EventProcessor . On ( mautrix . StateRoomAvatar , handler . HandleRoomMetadata )
bridge . EventProcessor . On ( mautrix . StateTopic , handler . HandleRoomMetadata )
2018-08-18 19:57:08 +00:00
return handler
}
2019-01-11 19:17:31 +00:00
func ( mx * MatrixHandler ) HandleBotInvite ( evt * mautrix . Event ) {
2018-08-18 19:57:08 +00:00
intent := mx . as . BotIntent ( )
2018-08-16 12:59:18 +00:00
2018-08-28 21:40:54 +00:00
user := mx . bridge . GetUserByMXID ( evt . Sender )
2018-08-26 14:08:37 +00:00
if user == nil {
return
}
2018-08-16 16:20:07 +00:00
resp , err := intent . JoinRoom ( evt . RoomID , "" , nil )
2018-08-16 12:59:18 +00:00
if err != nil {
2018-08-18 19:57:08 +00:00
mx . log . Debugln ( "Failed to join room" , evt . RoomID , "with invite from" , evt . Sender )
2018-08-16 12:59:18 +00:00
return
}
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 {
2018-08-18 19:57:08 +00:00
mx . log . Debugln ( "Failed to get members in room" , resp . RoomID , "after accepting invite from" , evt . Sender )
2018-08-16 16:20:07 +00:00
intent . LeaveRoom ( resp . RoomID )
2018-08-16 12:59:18 +00:00
return
}
if len ( members . Joined ) < 2 {
2018-08-18 19:57:08 +00:00
mx . log . Debugln ( "Leaving empty room" , resp . RoomID , "after accepting invite from" , evt . Sender )
2018-08-16 16:20:07 +00:00
intent . LeaveRoom ( resp . RoomID )
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 {
intent . SendNotice ( resp . RoomID , "You are not whitelisted to use this bridge.\n" +
"If you're the owner of this bridge, see the bridge.permissions section in your config file." )
intent . LeaveRoom ( resp . RoomID )
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
2018-08-28 21:40:54 +00:00
} else if _ , ok := mx . bridge . ParsePuppetMXID ( types . MatrixUserID ( mxid ) ) ; ok {
2018-08-16 21:11:28 +00:00
hasPuppets = true
2018-08-16 12:59:18 +00:00
continue
}
2018-08-18 19:57:08 +00:00
mx . log . Debugln ( "Leaving multi-user room" , resp . RoomID , "after accepting invite from" , evt . Sender )
2018-08-16 16:20:07 +00:00
intent . SendNotice ( resp . RoomID , "This bridge is user-specific, please don't invite me into rooms with other users." )
intent . LeaveRoom ( resp . RoomID )
2018-08-16 12:59:18 +00:00
return
}
2018-08-16 21:11:28 +00:00
if ! hasPuppets {
2018-08-28 21:40:54 +00:00
user := mx . bridge . GetUserByMXID ( types . MatrixUserID ( evt . Sender ) )
2018-08-18 19:57:08 +00:00
user . SetManagementRoom ( types . MatrixRoomID ( resp . RoomID ) )
2018-12-07 13:31:55 +00:00
intent . SendNotice ( string ( user . ManagementRoom ) , "This room has been registered as your bridge management/status room. Send `help` to get a list of commands." )
2018-08-18 19:57:08 +00:00
mx . log . Debugln ( resp . RoomID , "registered as a management room with" , evt . Sender )
2018-08-16 12:59:18 +00:00
}
}
2019-01-11 19:17:31 +00:00
func ( mx * MatrixHandler ) HandleMembership ( evt * mautrix . Event ) {
2018-08-18 19:57:08 +00:00
if evt . Content . Membership == "invite" && evt . GetStateKey ( ) == mx . as . BotMXID ( ) {
mx . HandleBotInvite ( evt )
2018-08-16 21:11:28 +00:00
}
2018-08-16 12:59:18 +00:00
}
2019-01-11 19:17:31 +00:00
func ( mx * MatrixHandler ) HandleRoomMetadata ( evt * mautrix . Event ) {
2018-08-28 21:40:54 +00:00
user := mx . bridge . GetUserByMXID ( types . MatrixUserID ( evt . Sender ) )
if user == nil || ! user . Whitelisted || ! user . IsLoggedIn ( ) {
2018-08-26 14:02:32 +00:00
return
}
2018-08-28 21:40:54 +00:00
portal := mx . bridge . GetPortalByMXID ( evt . RoomID )
2018-08-26 14:02:32 +00:00
if portal == nil || portal . IsPrivateChat ( ) {
return
}
var resp <- chan string
var err error
switch evt . Type {
2019-01-11 19:17:31 +00:00
case mautrix . StateRoomName :
2018-08-28 21:40:54 +00:00
resp , err = user . Conn . UpdateGroupSubject ( evt . Content . Name , portal . Key . JID )
2019-01-11 19:17:31 +00:00
case mautrix . StateRoomAvatar :
2018-08-26 14:02:32 +00:00
return
2019-01-11 19:17:31 +00:00
case mautrix . StateTopic :
2018-08-26 14:02:32 +00:00
return
}
if err != nil {
mx . log . Errorln ( err )
} else {
out := <- resp
mx . log . Infoln ( out )
}
}
2019-01-11 19:17:31 +00:00
func ( mx * MatrixHandler ) HandleMessage ( evt * mautrix . Event ) {
2018-09-01 20:38:03 +00:00
if _ , isPuppet := mx . bridge . ParsePuppetMXID ( evt . Sender ) ; evt . Sender == mx . bridge . Bot . UserID || isPuppet {
return
}
2018-08-18 19:57:08 +00:00
roomID := types . MatrixRoomID ( evt . RoomID )
2018-08-28 21:40:54 +00:00
user := mx . bridge . GetUserByMXID ( types . MatrixUserID ( evt . Sender ) )
2018-08-18 19:57:08 +00:00
2018-08-26 14:08:37 +00:00
if ! user . Whitelisted {
return
}
2019-01-11 19:17:31 +00:00
if evt . Content . MsgType == mautrix . MsgText {
2018-08-18 19:57:08 +00:00
commandPrefix := mx . bridge . Config . Bridge . CommandPrefix
hasCommandPrefix := strings . HasPrefix ( evt . Content . Body , commandPrefix )
if hasCommandPrefix {
evt . Content . Body = strings . TrimLeft ( evt . Content . Body [ len ( commandPrefix ) : ] , " " )
}
if hasCommandPrefix || roomID == user . ManagementRoom {
mx . cmd . Handle ( roomID , user , evt . Content . Body )
return
}
}
2018-08-28 21:40:54 +00:00
if ! user . IsLoggedIn ( ) {
return
}
portal := mx . bridge . GetPortalByMXID ( roomID )
2018-08-18 19:57:08 +00:00
if portal != nil {
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
}