user.go 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286
  1. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
  2. // Copyright (C) 2020 Tulir Asokan
  3. //
  4. // This program is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU Affero General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // This program is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU Affero General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU Affero General Public License
  15. // along with this program. If not, see <https://www.gnu.org/licenses/>.
  16. package main
  17. import (
  18. "context"
  19. "encoding/json"
  20. "errors"
  21. "fmt"
  22. "net/http"
  23. "sort"
  24. "strconv"
  25. "strings"
  26. "sync"
  27. "time"
  28. "github.com/skip2/go-qrcode"
  29. log "maunium.net/go/maulogger/v2"
  30. "github.com/Rhymen/go-whatsapp"
  31. waBinary "github.com/Rhymen/go-whatsapp/binary"
  32. waProto "github.com/Rhymen/go-whatsapp/binary/proto"
  33. "maunium.net/go/mautrix"
  34. "maunium.net/go/mautrix/event"
  35. "maunium.net/go/mautrix/format"
  36. "maunium.net/go/mautrix/id"
  37. "maunium.net/go/mautrix-whatsapp/database"
  38. )
  39. type User struct {
  40. *database.User
  41. Conn *whatsapp.Conn
  42. bridge *Bridge
  43. log log.Logger
  44. Admin bool
  45. Whitelisted bool
  46. RelaybotWhitelisted bool
  47. IsRelaybot bool
  48. ConnectionErrors int
  49. CommunityID string
  50. cleanDisconnection bool
  51. batteryWarningsSent int
  52. lastReconnection int64
  53. pushName string
  54. chatListReceived chan struct{}
  55. syncPortalsDone chan struct{}
  56. messageInput chan PortalMessage
  57. messageOutput chan PortalMessage
  58. syncStart chan struct{}
  59. syncWait sync.WaitGroup
  60. mgmtCreateLock sync.Mutex
  61. connLock sync.Mutex
  62. cancelReconnect func()
  63. prevBridgeStatus *AsmuxPong
  64. }
  65. func (bridge *Bridge) GetUserByMXID(userID id.UserID) *User {
  66. _, isPuppet := bridge.ParsePuppetMXID(userID)
  67. if isPuppet || userID == bridge.Bot.UserID {
  68. return nil
  69. }
  70. bridge.usersLock.Lock()
  71. defer bridge.usersLock.Unlock()
  72. user, ok := bridge.usersByMXID[userID]
  73. if !ok {
  74. return bridge.loadDBUser(bridge.DB.User.GetByMXID(userID), &userID)
  75. }
  76. return user
  77. }
  78. func (bridge *Bridge) GetUserByJID(userID whatsapp.JID) *User {
  79. bridge.usersLock.Lock()
  80. defer bridge.usersLock.Unlock()
  81. user, ok := bridge.usersByJID[userID]
  82. if !ok {
  83. return bridge.loadDBUser(bridge.DB.User.GetByJID(userID), nil)
  84. }
  85. return user
  86. }
  87. func (user *User) addToJIDMap() {
  88. user.bridge.usersLock.Lock()
  89. user.bridge.usersByJID[user.JID] = user
  90. user.bridge.usersLock.Unlock()
  91. }
  92. func (user *User) removeFromJIDMap() {
  93. user.bridge.usersLock.Lock()
  94. jidUser, ok := user.bridge.usersByJID[user.JID]
  95. if ok && user == jidUser {
  96. delete(user.bridge.usersByJID, user.JID)
  97. }
  98. user.bridge.usersLock.Unlock()
  99. user.bridge.Metrics.TrackLoginState(user.JID, false)
  100. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotLoggedIn})
  101. }
  102. func (bridge *Bridge) GetAllUsers() []*User {
  103. bridge.usersLock.Lock()
  104. defer bridge.usersLock.Unlock()
  105. dbUsers := bridge.DB.User.GetAll()
  106. output := make([]*User, len(dbUsers))
  107. for index, dbUser := range dbUsers {
  108. user, ok := bridge.usersByMXID[dbUser.MXID]
  109. if !ok {
  110. user = bridge.loadDBUser(dbUser, nil)
  111. }
  112. output[index] = user
  113. }
  114. return output
  115. }
  116. func (bridge *Bridge) loadDBUser(dbUser *database.User, mxid *id.UserID) *User {
  117. if dbUser == nil {
  118. if mxid == nil {
  119. return nil
  120. }
  121. dbUser = bridge.DB.User.New()
  122. dbUser.MXID = *mxid
  123. dbUser.Insert()
  124. }
  125. user := bridge.NewUser(dbUser)
  126. bridge.usersByMXID[user.MXID] = user
  127. if len(user.JID) > 0 {
  128. bridge.usersByJID[user.JID] = user
  129. }
  130. if len(user.ManagementRoom) > 0 {
  131. bridge.managementRooms[user.ManagementRoom] = user
  132. }
  133. return user
  134. }
  135. func (user *User) GetPortals() []*Portal {
  136. keys := user.User.GetPortalKeys()
  137. portals := make([]*Portal, len(keys))
  138. user.bridge.portalsLock.Lock()
  139. for i, key := range keys {
  140. portal, ok := user.bridge.portalsByJID[key]
  141. if !ok {
  142. portal = user.bridge.loadDBPortal(user.bridge.DB.Portal.GetByJID(key), &key)
  143. }
  144. portals[i] = portal
  145. }
  146. user.bridge.portalsLock.Unlock()
  147. return portals
  148. }
  149. func (bridge *Bridge) NewUser(dbUser *database.User) *User {
  150. user := &User{
  151. User: dbUser,
  152. bridge: bridge,
  153. log: bridge.Log.Sub("User").Sub(string(dbUser.MXID)),
  154. IsRelaybot: false,
  155. chatListReceived: make(chan struct{}, 1),
  156. syncPortalsDone: make(chan struct{}, 1),
  157. syncStart: make(chan struct{}, 1),
  158. messageInput: make(chan PortalMessage),
  159. messageOutput: make(chan PortalMessage, bridge.Config.Bridge.UserMessageBuffer),
  160. }
  161. user.RelaybotWhitelisted = user.bridge.Config.Bridge.Permissions.IsRelaybotWhitelisted(user.MXID)
  162. user.Whitelisted = user.bridge.Config.Bridge.Permissions.IsWhitelisted(user.MXID)
  163. user.Admin = user.bridge.Config.Bridge.Permissions.IsAdmin(user.MXID)
  164. go user.handleMessageLoop()
  165. go user.runMessageRingBuffer()
  166. return user
  167. }
  168. func (user *User) GetManagementRoom() id.RoomID {
  169. if len(user.ManagementRoom) == 0 {
  170. user.mgmtCreateLock.Lock()
  171. defer user.mgmtCreateLock.Unlock()
  172. if len(user.ManagementRoom) > 0 {
  173. return user.ManagementRoom
  174. }
  175. resp, err := user.bridge.Bot.CreateRoom(&mautrix.ReqCreateRoom{
  176. Topic: "WhatsApp bridge notices",
  177. IsDirect: true,
  178. })
  179. if err != nil {
  180. user.log.Errorln("Failed to auto-create management room:", err)
  181. } else {
  182. user.SetManagementRoom(resp.RoomID)
  183. }
  184. }
  185. return user.ManagementRoom
  186. }
  187. func (user *User) SetManagementRoom(roomID id.RoomID) {
  188. existingUser, ok := user.bridge.managementRooms[roomID]
  189. if ok {
  190. existingUser.ManagementRoom = ""
  191. existingUser.Update()
  192. }
  193. user.ManagementRoom = roomID
  194. user.bridge.managementRooms[user.ManagementRoom] = user
  195. user.Update()
  196. }
  197. func (user *User) SetSession(session *whatsapp.Session) {
  198. if session == nil {
  199. user.Session = nil
  200. user.LastConnection = 0
  201. } else if len(session.Wid) > 0 {
  202. user.Session = session
  203. } else {
  204. return
  205. }
  206. user.Update()
  207. }
  208. func (user *User) Connect(evenIfNoSession bool) bool {
  209. user.connLock.Lock()
  210. if user.Conn != nil {
  211. user.connLock.Unlock()
  212. if user.Conn.IsConnected() {
  213. return true
  214. } else {
  215. return user.RestoreSession()
  216. }
  217. } else if !evenIfNoSession && user.Session == nil {
  218. user.connLock.Unlock()
  219. return false
  220. }
  221. user.log.Debugln("Connecting to WhatsApp")
  222. if user.Session != nil {
  223. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWAConnecting})
  224. }
  225. timeout := time.Duration(user.bridge.Config.Bridge.ConnectionTimeout)
  226. if timeout == 0 {
  227. timeout = 20
  228. }
  229. user.Conn = whatsapp.NewConn(&whatsapp.Options{
  230. Timeout: timeout * time.Second,
  231. LongClientName: user.bridge.Config.WhatsApp.OSName,
  232. ShortClientName: user.bridge.Config.WhatsApp.BrowserName,
  233. ClientVersion: WAVersion,
  234. Log: user.log.Sub("Conn"),
  235. Handler: []whatsapp.Handler{user},
  236. })
  237. user.setupAdminTestHooks()
  238. user.connLock.Unlock()
  239. return user.RestoreSession()
  240. }
  241. func (user *User) DeleteConnection() {
  242. user.connLock.Lock()
  243. if user.Conn == nil {
  244. user.connLock.Unlock()
  245. return
  246. }
  247. err := user.Conn.Disconnect()
  248. if err != nil && err != whatsapp.ErrNotConnected {
  249. user.log.Warnln("Error disconnecting: %v", err)
  250. }
  251. user.Conn.RemoveHandlers()
  252. user.Conn = nil
  253. user.bridge.Metrics.TrackConnectionState(user.JID, false)
  254. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  255. user.connLock.Unlock()
  256. }
  257. func (user *User) RestoreSession() bool {
  258. if user.Session != nil {
  259. user.Conn.SetSession(*user.Session)
  260. ctx, cancel := context.WithTimeout(context.Background(), 3*time.Minute)
  261. defer cancel()
  262. err := user.Conn.Restore(true, ctx)
  263. if err == whatsapp.ErrAlreadyLoggedIn {
  264. return true
  265. } else if err != nil {
  266. user.log.Errorln("Failed to restore session:", err)
  267. if errors.Is(err, whatsapp.ErrUnpaired) {
  268. user.sendMarkdownBridgeAlert("\u26a0 Failed to connect to WhatsApp: unpaired from phone. " +
  269. "To re-pair your phone, log in again.")
  270. user.removeFromJIDMap()
  271. //user.JID = ""
  272. user.SetSession(nil)
  273. user.DeleteConnection()
  274. return false
  275. } else {
  276. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  277. user.sendMarkdownBridgeAlert("\u26a0 Failed to connect to WhatsApp. Make sure WhatsApp " +
  278. "on your phone is reachable and use `reconnect` to try connecting again.")
  279. }
  280. user.log.Debugln("Disconnecting due to failed session restore...")
  281. err = user.Conn.Disconnect()
  282. if err != nil {
  283. user.log.Errorln("Failed to disconnect after failed session restore:", err)
  284. }
  285. return false
  286. }
  287. user.ConnectionErrors = 0
  288. user.log.Debugln("Session restored successfully")
  289. user.PostLogin()
  290. }
  291. return true
  292. }
  293. func (user *User) HasSession() bool {
  294. return user.Session != nil
  295. }
  296. func (user *User) IsConnected() bool {
  297. return user.Conn != nil && user.Conn.IsConnected() && user.Conn.IsLoggedIn()
  298. }
  299. func (user *User) IsLoginInProgress() bool {
  300. return user.Conn != nil && user.Conn.IsLoginInProgress()
  301. }
  302. func (user *User) loginQrChannel(ce *CommandEvent, qrChan <-chan string, eventIDChan chan<- id.EventID) {
  303. var qrEventID id.EventID
  304. for code := range qrChan {
  305. if code == "stop" {
  306. return
  307. }
  308. qrCode, err := qrcode.Encode(code, qrcode.Low, 256)
  309. if err != nil {
  310. user.log.Errorln("Failed to encode QR code:", err)
  311. ce.Reply("Failed to encode QR code: %v", err)
  312. return
  313. }
  314. bot := user.bridge.AS.BotClient()
  315. resp, err := bot.UploadBytes(qrCode, "image/png")
  316. if err != nil {
  317. user.log.Errorln("Failed to upload QR code:", err)
  318. ce.Reply("Failed to upload QR code: %v", err)
  319. return
  320. }
  321. if qrEventID == "" {
  322. sendResp, err := bot.SendImage(ce.RoomID, code, resp.ContentURI)
  323. if err != nil {
  324. user.log.Errorln("Failed to send QR code to user:", err)
  325. return
  326. }
  327. qrEventID = sendResp.EventID
  328. eventIDChan <- qrEventID
  329. } else {
  330. _, err = bot.SendMessageEvent(ce.RoomID, event.EventMessage, &event.MessageEventContent{
  331. MsgType: event.MsgImage,
  332. Body: code,
  333. URL: resp.ContentURI.CUString(),
  334. NewContent: &event.MessageEventContent{
  335. MsgType: event.MsgImage,
  336. Body: code,
  337. URL: resp.ContentURI.CUString(),
  338. },
  339. RelatesTo: &event.RelatesTo{
  340. Type: event.RelReplace,
  341. EventID: qrEventID,
  342. },
  343. })
  344. if err != nil {
  345. user.log.Errorln("Failed to send edited QR code to user:", err)
  346. }
  347. }
  348. }
  349. }
  350. func (user *User) Login(ce *CommandEvent) {
  351. qrChan := make(chan string, 3)
  352. eventIDChan := make(chan id.EventID, 1)
  353. go user.loginQrChannel(ce, qrChan, eventIDChan)
  354. session, jid, err := user.Conn.Login(qrChan, nil, user.bridge.Config.Bridge.LoginQRRegenCount)
  355. qrChan <- "stop"
  356. if err != nil {
  357. var eventID id.EventID
  358. select {
  359. case eventID = <-eventIDChan:
  360. default:
  361. }
  362. reply := event.MessageEventContent{
  363. MsgType: event.MsgText,
  364. }
  365. if err == whatsapp.ErrAlreadyLoggedIn {
  366. reply.Body = "You're already logged in"
  367. } else if err == whatsapp.ErrLoginInProgress {
  368. reply.Body = "You have a login in progress already."
  369. } else if err == whatsapp.ErrLoginTimedOut {
  370. reply.Body = "QR code scan timed out. Please try again."
  371. } else {
  372. user.log.Warnln("Failed to log in:", err)
  373. reply.Body = fmt.Sprintf("Unknown error while logging in: %v", err)
  374. }
  375. msg := reply
  376. if eventID != "" {
  377. msg.NewContent = &reply
  378. msg.RelatesTo = &event.RelatesTo{
  379. Type: event.RelReplace,
  380. EventID: eventID,
  381. }
  382. }
  383. _, _ = ce.Bot.SendMessageEvent(ce.RoomID, event.EventMessage, &msg)
  384. return
  385. }
  386. // TODO there's a bit of duplication between this and the provisioning API login method
  387. // Also between the two logout methods (commands.go and provisioning.go)
  388. user.log.Debugln("Successful login as", jid, "via command")
  389. user.ConnectionErrors = 0
  390. user.JID = strings.Replace(jid, whatsapp.OldUserSuffix, whatsapp.NewUserSuffix, 1)
  391. user.addToJIDMap()
  392. user.SetSession(&session)
  393. ce.Reply("Successfully logged in, synchronizing chats...")
  394. user.PostLogin()
  395. }
  396. type Chat struct {
  397. Portal *Portal
  398. LastMessageTime uint64
  399. MarkAsRead bool
  400. Contact whatsapp.Contact
  401. }
  402. type ChatList []Chat
  403. func (cl ChatList) Len() int {
  404. return len(cl)
  405. }
  406. func (cl ChatList) Less(i, j int) bool {
  407. return cl[i].LastMessageTime > cl[j].LastMessageTime
  408. }
  409. func (cl ChatList) Swap(i, j int) {
  410. cl[i], cl[j] = cl[j], cl[i]
  411. }
  412. func (user *User) PostLogin() {
  413. user.sendBridgeStatus(AsmuxPong{OK: true})
  414. user.bridge.Metrics.TrackConnectionState(user.JID, true)
  415. user.bridge.Metrics.TrackLoginState(user.JID, true)
  416. user.bridge.Metrics.TrackBufferLength(user.MXID, 0)
  417. user.log.Debugln("Locking processing of incoming messages and starting post-login sync")
  418. // TODO can an old sync still be ongoing when PostLogin is called again?
  419. // TODO 2: can the new sync happen before this happens?
  420. user.chatListReceived = make(chan struct{}, 1)
  421. user.syncPortalsDone = make(chan struct{}, 1)
  422. user.syncWait.Add(1)
  423. user.syncStart <- struct{}{}
  424. go user.intPostLogin()
  425. }
  426. func (user *User) tryAutomaticDoublePuppeting() {
  427. if !user.bridge.Config.CanDoublePuppet(user.MXID) {
  428. return
  429. }
  430. user.log.Debugln("Checking if double puppeting needs to be enabled")
  431. puppet := user.bridge.GetPuppetByJID(user.JID)
  432. if len(puppet.CustomMXID) > 0 {
  433. user.log.Debugln("User already has double-puppeting enabled")
  434. // Custom puppet already enabled
  435. return
  436. }
  437. accessToken, err := puppet.loginWithSharedSecret(user.MXID)
  438. if err != nil {
  439. user.log.Warnln("Failed to login with shared secret:", err)
  440. return
  441. }
  442. err = puppet.SwitchCustomMXID(accessToken, user.MXID)
  443. if err != nil {
  444. puppet.log.Warnln("Failed to switch to auto-logined custom puppet:", err)
  445. return
  446. }
  447. user.log.Infoln("Successfully automatically enabled custom puppet")
  448. }
  449. func (user *User) sendBridgeNotice(formatString string, args ...interface{}) {
  450. notice := fmt.Sprintf(formatString, args...)
  451. _, err := user.bridge.Bot.SendNotice(user.GetManagementRoom(), notice)
  452. if err != nil {
  453. user.log.Warnf("Failed to send bridge notice \"%s\": %v", notice, err)
  454. }
  455. }
  456. func (user *User) sendMarkdownBridgeAlert(formatString string, args ...interface{}) {
  457. notice := fmt.Sprintf(formatString, args...)
  458. content := format.RenderMarkdown(notice, true, false)
  459. _, err := user.bridge.Bot.SendMessageEvent(user.GetManagementRoom(), event.EventMessage, content)
  460. if err != nil {
  461. user.log.Warnf("Failed to send bridge alert \"%s\": %v", notice, err)
  462. }
  463. }
  464. func (user *User) postConnPing() bool {
  465. user.log.Debugln("Making post-connection ping")
  466. var err error
  467. for i := 0; ; i++ {
  468. err = user.Conn.AdminTest()
  469. if err == nil {
  470. user.log.Debugln("Post-connection ping OK")
  471. return true
  472. } else if errors.Is(err, whatsapp.ErrConnectionTimeout) && i < 5 {
  473. user.log.Warnfln("Post-connection ping timed out, sending new one")
  474. } else {
  475. break
  476. }
  477. }
  478. user.log.Errorfln("Post-connection ping failed: %v. Disconnecting and then reconnecting after a second", err)
  479. disconnectErr := user.Conn.Disconnect()
  480. if disconnectErr != nil {
  481. user.log.Warnln("Error while disconnecting after failed post-connection ping:", disconnectErr)
  482. }
  483. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  484. user.bridge.Metrics.TrackDisconnection(user.MXID)
  485. go func() {
  486. time.Sleep(1 * time.Second)
  487. user.tryReconnect(fmt.Sprintf("Post-connection ping failed: %v", err))
  488. }()
  489. return false
  490. }
  491. func (user *User) intPostLogin() {
  492. defer user.syncWait.Done()
  493. user.lastReconnection = time.Now().Unix()
  494. user.createCommunity()
  495. user.tryAutomaticDoublePuppeting()
  496. user.log.Debugln("Waiting for chat list receive confirmation")
  497. select {
  498. case <-user.chatListReceived:
  499. user.log.Debugln("Chat list receive confirmation received in PostLogin")
  500. case <-time.After(time.Duration(user.bridge.Config.Bridge.ChatListWait) * time.Second):
  501. user.log.Warnln("Timed out waiting for chat list to arrive!")
  502. user.postConnPing()
  503. return
  504. }
  505. if !user.postConnPing() {
  506. user.log.Debugln("Post-connection ping failed, unlocking processing of incoming messages.")
  507. return
  508. }
  509. user.log.Debugln("Waiting for portal sync complete confirmation")
  510. select {
  511. case <-user.syncPortalsDone:
  512. user.log.Debugln("Post-connection portal sync complete, unlocking processing of incoming messages.")
  513. // TODO this is too short, maybe a per-portal duration?
  514. case <-time.After(time.Duration(user.bridge.Config.Bridge.PortalSyncWait) * time.Second):
  515. user.log.Warnln("Timed out waiting for portal sync to complete! Unlocking processing of incoming messages.")
  516. }
  517. }
  518. type NormalMessage interface {
  519. GetInfo() whatsapp.MessageInfo
  520. }
  521. func (user *User) HandleEvent(event interface{}) {
  522. switch v := event.(type) {
  523. case NormalMessage:
  524. info := v.GetInfo()
  525. user.messageInput <- PortalMessage{info.RemoteJid, user, v, info.Timestamp}
  526. case whatsapp.MessageRevocation:
  527. user.messageInput <- PortalMessage{v.RemoteJid, user, v, 0}
  528. case whatsapp.StreamEvent:
  529. user.HandleStreamEvent(v)
  530. case []whatsapp.Chat:
  531. user.HandleChatList(v)
  532. case []whatsapp.Contact:
  533. user.HandleContactList(v)
  534. case error:
  535. user.HandleError(v)
  536. case whatsapp.Contact:
  537. go user.HandleNewContact(v)
  538. case whatsapp.BatteryMessage:
  539. user.HandleBatteryMessage(v)
  540. case whatsapp.CallInfo:
  541. user.HandleCallInfo(v)
  542. case whatsapp.PresenceEvent:
  543. go user.HandlePresence(v)
  544. case whatsapp.JSONMsgInfo:
  545. go user.HandleMsgInfo(v)
  546. case whatsapp.ReceivedMessage:
  547. user.HandleReceivedMessage(v)
  548. case whatsapp.ReadMessage:
  549. user.HandleReadMessage(v)
  550. case whatsapp.JSONCommand:
  551. user.HandleCommand(v)
  552. case whatsapp.ChatUpdate:
  553. user.HandleChatUpdate(v)
  554. case whatsapp.ConnInfo:
  555. user.HandleConnInfo(v)
  556. case json.RawMessage:
  557. user.HandleJSONMessage(v)
  558. case *waProto.WebMessageInfo:
  559. user.updateLastConnectionIfNecessary()
  560. // TODO trace log
  561. //user.log.Debugfln("WebMessageInfo: %+v", v)
  562. case *waBinary.Node:
  563. user.log.Debugfln("Unknown binary message: %+v", v)
  564. default:
  565. user.log.Debugfln("Unknown type of event in HandleEvent: %T", v)
  566. }
  567. }
  568. func (user *User) HandleStreamEvent(evt whatsapp.StreamEvent) {
  569. if evt.Type == whatsapp.StreamSleep {
  570. if user.lastReconnection+60 > time.Now().Unix() {
  571. user.lastReconnection = 0
  572. user.log.Infoln("Stream went to sleep soon after reconnection, making new post-connection ping in 20 seconds")
  573. go func() {
  574. time.Sleep(20 * time.Second)
  575. // TODO if this happens during the post-login sync, it can get stuck forever
  576. // TODO check if the above is still true
  577. user.postConnPing()
  578. }()
  579. }
  580. } else {
  581. user.log.Infofln("Stream event: %+v", evt)
  582. }
  583. }
  584. func (user *User) HandleChatList(chats []whatsapp.Chat) {
  585. user.log.Infoln("Chat list received")
  586. chatMap := make(map[string]whatsapp.Chat)
  587. for _, chat := range user.Conn.Store.Chats {
  588. chatMap[chat.JID] = chat
  589. }
  590. for _, chat := range chats {
  591. chatMap[chat.JID] = chat
  592. }
  593. select {
  594. case user.chatListReceived <- struct{}{}:
  595. default:
  596. }
  597. go user.syncPortals(chatMap, false)
  598. }
  599. func (user *User) syncPortals(chatMap map[string]whatsapp.Chat, createAll bool) {
  600. // TODO use contexts instead of checking if user.Conn is the same?
  601. connAtStart := user.Conn
  602. if chatMap == nil {
  603. chatMap = user.Conn.Store.Chats
  604. }
  605. user.log.Infoln("Reading chat list")
  606. chats := make(ChatList, 0, len(chatMap))
  607. existingKeys := user.GetInCommunityMap()
  608. portalKeys := make([]database.PortalKeyWithMeta, 0, len(chatMap))
  609. for _, chat := range chatMap {
  610. ts, err := strconv.ParseUint(chat.LastMessageTime, 10, 64)
  611. if err != nil {
  612. user.log.Warnfln("Non-integer last message time in %s: %s", chat.JID, chat.LastMessageTime)
  613. continue
  614. }
  615. portal := user.GetPortalByJID(chat.JID)
  616. chats = append(chats, Chat{
  617. Portal: portal,
  618. Contact: user.Conn.Store.Contacts[chat.JID],
  619. LastMessageTime: ts,
  620. MarkAsRead: chat.Unread == "0",
  621. })
  622. var inCommunity, ok bool
  623. if inCommunity, ok = existingKeys[portal.Key]; !ok || !inCommunity {
  624. inCommunity = user.addPortalToCommunity(portal)
  625. if portal.IsPrivateChat() {
  626. puppet := user.bridge.GetPuppetByJID(portal.Key.JID)
  627. user.addPuppetToCommunity(puppet)
  628. }
  629. }
  630. portalKeys = append(portalKeys, database.PortalKeyWithMeta{PortalKey: portal.Key, InCommunity: inCommunity})
  631. }
  632. user.log.Infoln("Read chat list, updating user-portal mapping")
  633. err := user.SetPortalKeys(portalKeys)
  634. if err != nil {
  635. user.log.Warnln("Failed to update user-portal mapping:", err)
  636. }
  637. sort.Sort(chats)
  638. limit := user.bridge.Config.Bridge.InitialChatSync
  639. if limit < 0 {
  640. limit = len(chats)
  641. }
  642. if user.Conn != connAtStart {
  643. user.log.Debugln("Connection seems to have changed before sync, cancelling")
  644. return
  645. }
  646. now := uint64(time.Now().Unix())
  647. user.log.Infoln("Syncing portals")
  648. doublePuppet := user.bridge.GetPuppetByCustomMXID(user.MXID)
  649. for i, chat := range chats {
  650. if chat.LastMessageTime+user.bridge.Config.Bridge.SyncChatMaxAge < now {
  651. break
  652. }
  653. create := (chat.LastMessageTime >= user.LastConnection && user.LastConnection > 0) || i < limit
  654. if len(chat.Portal.MXID) > 0 || create || createAll {
  655. // Don't sync unless chat meta sync is enabled or portal doesn't exist
  656. if user.bridge.Config.Bridge.ChatMetaSync || len(chat.Portal.MXID) == 0 {
  657. chat.Portal.Sync(user, chat.Contact)
  658. }
  659. err = chat.Portal.BackfillHistory(user, chat.LastMessageTime)
  660. if err != nil {
  661. chat.Portal.log.Errorln("Error backfilling history:", err)
  662. }
  663. if chat.MarkAsRead && doublePuppet != nil && doublePuppet.CustomIntent() != nil {
  664. lastMessage := user.bridge.DB.Message.GetLastInChat(chat.Portal.Key)
  665. if lastMessage != nil {
  666. err = doublePuppet.CustomIntent().MarkRead(chat.Portal.MXID, lastMessage.MXID)
  667. if err != nil {
  668. user.log.Warnln("Failed to mark %s in %s as read after backfill: %v", lastMessage.MXID, chat.Portal.MXID, err)
  669. }
  670. }
  671. }
  672. }
  673. }
  674. if user.Conn != connAtStart {
  675. user.log.Debugln("Connection seems to have changed during sync, cancelling")
  676. return
  677. }
  678. user.UpdateDirectChats(nil)
  679. user.log.Infoln("Finished syncing portals")
  680. select {
  681. case user.syncPortalsDone <- struct{}{}:
  682. default:
  683. }
  684. }
  685. func (user *User) getDirectChats() map[id.UserID][]id.RoomID {
  686. res := make(map[id.UserID][]id.RoomID)
  687. privateChats := user.bridge.DB.Portal.FindPrivateChats(user.JID)
  688. for _, portal := range privateChats {
  689. if len(portal.MXID) > 0 {
  690. res[user.bridge.FormatPuppetMXID(portal.Key.JID)] = []id.RoomID{portal.MXID}
  691. }
  692. }
  693. return res
  694. }
  695. func (user *User) UpdateDirectChats(chats map[id.UserID][]id.RoomID) {
  696. if !user.bridge.Config.Bridge.SyncDirectChatList {
  697. return
  698. }
  699. puppet := user.bridge.GetPuppetByCustomMXID(user.MXID)
  700. if puppet == nil || puppet.CustomIntent() == nil {
  701. return
  702. }
  703. intent := puppet.CustomIntent()
  704. method := http.MethodPatch
  705. if chats == nil {
  706. chats = user.getDirectChats()
  707. method = http.MethodPut
  708. }
  709. user.log.Debugln("Updating m.direct list on homeserver")
  710. var err error
  711. if user.bridge.Config.Homeserver.Asmux {
  712. urlPath := intent.BuildBaseURL("_matrix", "client", "unstable", "com.beeper.asmux", "dms")
  713. _, err = intent.MakeFullRequest(mautrix.FullRequest{
  714. Method: method,
  715. URL: urlPath,
  716. Headers: http.Header{"X-Asmux-Auth": {user.bridge.AS.Registration.AppToken}},
  717. RequestJSON: chats,
  718. })
  719. } else {
  720. existingChats := make(map[id.UserID][]id.RoomID)
  721. err = intent.GetAccountData(event.AccountDataDirectChats.Type, &existingChats)
  722. if err != nil {
  723. user.log.Warnln("Failed to get m.direct list to update it:", err)
  724. return
  725. }
  726. for userID, rooms := range existingChats {
  727. if _, ok := user.bridge.ParsePuppetMXID(userID); !ok {
  728. // This is not a ghost user, include it in the new list
  729. chats[userID] = rooms
  730. } else if _, ok := chats[userID]; !ok && method == http.MethodPatch {
  731. // This is a ghost user, but we're not replacing the whole list, so include it too
  732. chats[userID] = rooms
  733. }
  734. }
  735. err = intent.SetAccountData(event.AccountDataDirectChats.Type, &chats)
  736. }
  737. if err != nil {
  738. user.log.Warnln("Failed to update m.direct list:", err)
  739. }
  740. }
  741. func (user *User) HandleContactList(contacts []whatsapp.Contact) {
  742. contactMap := make(map[whatsapp.JID]whatsapp.Contact)
  743. for _, contact := range contacts {
  744. contactMap[contact.JID] = contact
  745. }
  746. go user.syncPuppets(contactMap)
  747. }
  748. func (user *User) syncPuppets(contacts map[whatsapp.JID]whatsapp.Contact) {
  749. if contacts == nil {
  750. contacts = user.Conn.Store.Contacts
  751. }
  752. _, hasSelf := contacts[user.JID]
  753. if !hasSelf {
  754. contacts[user.JID] = whatsapp.Contact{
  755. Name: user.pushName,
  756. Notify: user.pushName,
  757. JID: user.JID,
  758. }
  759. }
  760. user.log.Infoln("Syncing puppet info from contacts")
  761. for jid, contact := range contacts {
  762. if strings.HasSuffix(jid, whatsapp.NewUserSuffix) {
  763. puppet := user.bridge.GetPuppetByJID(contact.JID)
  764. puppet.Sync(user, contact)
  765. } else if strings.HasSuffix(jid, whatsapp.BroadcastSuffix) {
  766. portal := user.GetPortalByJID(contact.JID)
  767. portal.Sync(user, contact)
  768. }
  769. }
  770. user.log.Infoln("Finished syncing puppet info from contacts")
  771. }
  772. func (user *User) updateLastConnectionIfNecessary() {
  773. if user.LastConnection+60 < uint64(time.Now().Unix()) {
  774. user.UpdateLastConnection()
  775. }
  776. }
  777. func (user *User) HandleError(err error) {
  778. if !errors.Is(err, whatsapp.ErrInvalidWsData) {
  779. user.log.Errorfln("WhatsApp error: %v", err)
  780. }
  781. if closed, ok := err.(*whatsapp.ErrConnectionClosed); ok {
  782. user.bridge.Metrics.TrackDisconnection(user.MXID)
  783. if closed.Code == 1000 && user.cleanDisconnection {
  784. user.cleanDisconnection = false
  785. if !user.bridge.Config.Bridge.AggressiveReconnect {
  786. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  787. user.bridge.Metrics.TrackConnectionState(user.JID, false)
  788. user.log.Infoln("Clean disconnection by server")
  789. return
  790. } else {
  791. user.log.Debugln("Clean disconnection by server, but aggressive reconnection is enabled")
  792. }
  793. }
  794. go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection was closed with websocket status code %d", closed.Code))
  795. } else if failed, ok := err.(*whatsapp.ErrConnectionFailed); ok {
  796. disconnectErr := user.Conn.Disconnect()
  797. if disconnectErr != nil {
  798. user.log.Warnln("Failed to disconnect after connection fail:", disconnectErr)
  799. }
  800. user.bridge.Metrics.TrackDisconnection(user.MXID)
  801. user.ConnectionErrors++
  802. go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection failed: %v", failed.Err))
  803. } else if err == whatsapp.ErrPingFalse {
  804. disconnectErr := user.Conn.Disconnect()
  805. if disconnectErr != nil {
  806. user.log.Warnln("Failed to disconnect after failed ping:", disconnectErr)
  807. }
  808. user.bridge.Metrics.TrackDisconnection(user.MXID)
  809. user.ConnectionErrors++
  810. go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection failed: %v", err))
  811. }
  812. // Otherwise unknown error, probably mostly harmless
  813. }
  814. func (user *User) tryReconnect(msg string) {
  815. user.bridge.Metrics.TrackConnectionState(user.JID, false)
  816. if user.ConnectionErrors > user.bridge.Config.Bridge.MaxConnectionAttempts {
  817. user.sendMarkdownBridgeAlert("%s. Use the `reconnect` command to reconnect.", msg)
  818. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  819. return
  820. }
  821. if user.bridge.Config.Bridge.ReportConnectionRetry {
  822. user.sendBridgeNotice("%s. Reconnecting...", msg)
  823. // Don't want the same error to be repeated
  824. msg = ""
  825. }
  826. var tries uint
  827. var exponentialBackoff bool
  828. baseDelay := time.Duration(user.bridge.Config.Bridge.ConnectionRetryDelay)
  829. if baseDelay < 0 {
  830. exponentialBackoff = true
  831. baseDelay = -baseDelay + 1
  832. }
  833. delay := baseDelay
  834. ctx, cancel := context.WithCancel(context.Background())
  835. defer cancel()
  836. user.cancelReconnect = cancel
  837. for user.ConnectionErrors <= user.bridge.Config.Bridge.MaxConnectionAttempts {
  838. select {
  839. case <-ctx.Done():
  840. user.log.Debugln("tryReconnect context cancelled, aborting reconnection attempts")
  841. return
  842. default:
  843. }
  844. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWAConnecting})
  845. err := user.Conn.Restore(true, ctx)
  846. if err == nil {
  847. user.ConnectionErrors = 0
  848. if user.bridge.Config.Bridge.ReportConnectionRetry {
  849. user.sendBridgeNotice("Reconnected successfully")
  850. }
  851. user.PostLogin()
  852. return
  853. } else if errors.Is(err, whatsapp.ErrBadRequest) {
  854. user.log.Warnln("Got init 400 error when trying to reconnect, resetting connection...")
  855. err = user.Conn.Disconnect()
  856. if err != nil {
  857. user.log.Debugln("Error while disconnecting for connection reset:", err)
  858. }
  859. } else if errors.Is(err, whatsapp.ErrUnpaired) {
  860. user.log.Errorln("Got init 401 (unpaired) error when trying to reconnect, not retrying")
  861. user.removeFromJIDMap()
  862. //user.JID = ""
  863. user.SetSession(nil)
  864. user.DeleteConnection()
  865. user.sendMarkdownBridgeAlert("\u26a0 Failed to reconnect to WhatsApp: unpaired from phone. " +
  866. "To re-pair your phone, log in again.")
  867. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotLoggedIn})
  868. return
  869. } else if errors.Is(err, whatsapp.ErrAlreadyLoggedIn) {
  870. user.log.Warnln("Reconnection said we're already logged in, not trying anymore")
  871. return
  872. } else {
  873. user.log.Errorln("Error while trying to reconnect after disconnection:", err)
  874. }
  875. tries++
  876. user.ConnectionErrors++
  877. if user.ConnectionErrors <= user.bridge.Config.Bridge.MaxConnectionAttempts {
  878. if exponentialBackoff {
  879. delay = (1 << tries) + baseDelay
  880. }
  881. if user.bridge.Config.Bridge.ReportConnectionRetry {
  882. user.sendBridgeNotice("Reconnection attempt failed: %v. Retrying in %d seconds...", err, delay)
  883. }
  884. time.Sleep(delay * time.Second)
  885. }
  886. }
  887. user.sendBridgeStatus(AsmuxPong{Error: AsmuxWANotConnected})
  888. if user.bridge.Config.Bridge.ReportConnectionRetry {
  889. user.sendMarkdownBridgeAlert("%d reconnection attempts failed. Use the `reconnect` command to try to reconnect manually.", tries)
  890. } else {
  891. user.sendMarkdownBridgeAlert("\u26a0 %s. Additionally, %d reconnection attempts failed. Use the `reconnect` command to try to reconnect.", msg, tries)
  892. }
  893. }
  894. func (user *User) PortalKey(jid whatsapp.JID) database.PortalKey {
  895. return database.NewPortalKey(jid, user.JID)
  896. }
  897. func (user *User) GetPortalByJID(jid whatsapp.JID) *Portal {
  898. return user.bridge.GetPortalByJID(user.PortalKey(jid))
  899. }
  900. func (user *User) runMessageRingBuffer() {
  901. for msg := range user.messageInput {
  902. select {
  903. case user.messageOutput <- msg:
  904. user.bridge.Metrics.TrackBufferLength(user.MXID, len(user.messageOutput))
  905. default:
  906. dropped := <-user.messageOutput
  907. user.log.Warnln("Buffer is full, dropping message in", dropped.chat)
  908. user.messageOutput <- msg
  909. }
  910. }
  911. }
  912. func (user *User) handleMessageLoop() {
  913. for {
  914. select {
  915. case msg := <-user.messageOutput:
  916. user.bridge.Metrics.TrackBufferLength(user.MXID, len(user.messageOutput))
  917. user.GetPortalByJID(msg.chat).messages <- msg
  918. case <-user.syncStart:
  919. user.log.Debugln("Processing of incoming messages is locked")
  920. user.bridge.Metrics.TrackSyncLock(user.JID, true)
  921. user.syncWait.Wait()
  922. user.bridge.Metrics.TrackSyncLock(user.JID, false)
  923. user.log.Debugln("Processing of incoming messages unlocked")
  924. }
  925. }
  926. }
  927. func (user *User) HandleNewContact(contact whatsapp.Contact) {
  928. user.log.Debugfln("Contact message: %+v", contact)
  929. if strings.HasSuffix(contact.JID, whatsapp.OldUserSuffix) {
  930. contact.JID = strings.Replace(contact.JID, whatsapp.OldUserSuffix, whatsapp.NewUserSuffix, -1)
  931. }
  932. if strings.HasSuffix(contact.JID, whatsapp.NewUserSuffix) {
  933. puppet := user.bridge.GetPuppetByJID(contact.JID)
  934. puppet.UpdateName(user, contact)
  935. } else if strings.HasSuffix(contact.JID, whatsapp.BroadcastSuffix) {
  936. portal := user.GetPortalByJID(contact.JID)
  937. portal.UpdateName(contact.Name, "", nil, true)
  938. }
  939. }
  940. func (user *User) HandleBatteryMessage(battery whatsapp.BatteryMessage) {
  941. user.log.Debugfln("Battery message: %+v", battery)
  942. var notice string
  943. if !battery.Plugged && battery.Percentage < 15 && user.batteryWarningsSent < 1 {
  944. notice = fmt.Sprintf("Phone battery low (%d %% remaining)", battery.Percentage)
  945. user.batteryWarningsSent = 1
  946. } else if !battery.Plugged && battery.Percentage < 5 && user.batteryWarningsSent < 2 {
  947. notice = fmt.Sprintf("Phone battery very low (%d %% remaining)", battery.Percentage)
  948. user.batteryWarningsSent = 2
  949. } else if battery.Percentage > 15 || battery.Plugged {
  950. user.batteryWarningsSent = 0
  951. }
  952. if notice != "" {
  953. go user.sendBridgeNotice("%s", notice)
  954. }
  955. }
  956. type FakeMessage struct {
  957. Text string
  958. ID string
  959. Alert bool
  960. }
  961. func (user *User) HandleCallInfo(info whatsapp.CallInfo) {
  962. if info.Data != nil {
  963. return
  964. }
  965. data := FakeMessage{
  966. ID: info.ID,
  967. }
  968. switch info.Type {
  969. case whatsapp.CallOffer:
  970. if !user.bridge.Config.Bridge.CallNotices.Start {
  971. return
  972. }
  973. data.Text = "Incoming call"
  974. data.Alert = true
  975. case whatsapp.CallOfferVideo:
  976. if !user.bridge.Config.Bridge.CallNotices.Start {
  977. return
  978. }
  979. data.Text = "Incoming video call"
  980. data.Alert = true
  981. case whatsapp.CallTerminate:
  982. if !user.bridge.Config.Bridge.CallNotices.End {
  983. return
  984. }
  985. data.Text = "Call ended"
  986. data.ID += "E"
  987. default:
  988. return
  989. }
  990. portal := user.GetPortalByJID(info.From)
  991. if portal != nil {
  992. portal.messages <- PortalMessage{info.From, user, data, 0}
  993. }
  994. }
  995. func (user *User) HandlePresence(info whatsapp.PresenceEvent) {
  996. puppet := user.bridge.GetPuppetByJID(info.SenderJID)
  997. switch info.Status {
  998. case whatsapp.PresenceUnavailable:
  999. _ = puppet.DefaultIntent().SetPresence("offline")
  1000. case whatsapp.PresenceAvailable:
  1001. if len(puppet.typingIn) > 0 && puppet.typingAt+15 > time.Now().Unix() {
  1002. portal := user.bridge.GetPortalByMXID(puppet.typingIn)
  1003. _, _ = puppet.IntentFor(portal).UserTyping(puppet.typingIn, false, 0)
  1004. puppet.typingIn = ""
  1005. puppet.typingAt = 0
  1006. } else {
  1007. _ = puppet.DefaultIntent().SetPresence("online")
  1008. }
  1009. case whatsapp.PresenceComposing:
  1010. portal := user.GetPortalByJID(info.JID)
  1011. if len(puppet.typingIn) > 0 && puppet.typingAt+15 > time.Now().Unix() {
  1012. if puppet.typingIn == portal.MXID {
  1013. return
  1014. }
  1015. _, _ = puppet.IntentFor(portal).UserTyping(puppet.typingIn, false, 0)
  1016. }
  1017. puppet.typingIn = portal.MXID
  1018. puppet.typingAt = time.Now().Unix()
  1019. _, _ = puppet.IntentFor(portal).UserTyping(portal.MXID, true, 15*1000)
  1020. }
  1021. }
  1022. func (user *User) HandleMsgInfo(info whatsapp.JSONMsgInfo) {
  1023. if (info.Command == whatsapp.MsgInfoCommandAck || info.Command == whatsapp.MsgInfoCommandAcks) && info.Acknowledgement == whatsapp.AckMessageRead {
  1024. portal := user.GetPortalByJID(info.ToJID)
  1025. if len(portal.MXID) == 0 {
  1026. return
  1027. }
  1028. intent := user.bridge.GetPuppetByJID(info.SenderJID).IntentFor(portal)
  1029. for _, msgID := range info.IDs {
  1030. msg := user.bridge.DB.Message.GetByJID(portal.Key, msgID)
  1031. if msg == nil || msg.IsFakeMXID() {
  1032. continue
  1033. }
  1034. err := intent.MarkRead(portal.MXID, msg.MXID)
  1035. if err != nil {
  1036. user.log.Warnln("Failed to mark message %s as read by %s: %v", msg.MXID, info.SenderJID, err)
  1037. }
  1038. }
  1039. }
  1040. }
  1041. func (user *User) HandleReceivedMessage(received whatsapp.ReceivedMessage) {
  1042. if received.Type == "read" {
  1043. go user.markSelfRead(received.Jid, received.Index)
  1044. } else {
  1045. user.log.Debugfln("Unknown received message type: %+v", received)
  1046. }
  1047. }
  1048. func (user *User) HandleReadMessage(read whatsapp.ReadMessage) {
  1049. user.log.Debugfln("Received chat read message: %+v", read)
  1050. go user.markSelfRead(read.Jid, "")
  1051. }
  1052. func (user *User) markSelfRead(jid, messageID string) {
  1053. if strings.HasSuffix(jid, whatsapp.OldUserSuffix) {
  1054. jid = strings.Replace(jid, whatsapp.OldUserSuffix, whatsapp.NewUserSuffix, -1)
  1055. }
  1056. puppet := user.bridge.GetPuppetByJID(user.JID)
  1057. if puppet == nil {
  1058. return
  1059. }
  1060. intent := puppet.CustomIntent()
  1061. if intent == nil {
  1062. return
  1063. }
  1064. portal := user.GetPortalByJID(jid)
  1065. if portal == nil {
  1066. return
  1067. }
  1068. var message *database.Message
  1069. if messageID == "" {
  1070. message = user.bridge.DB.Message.GetLastInChat(portal.Key)
  1071. if message == nil {
  1072. return
  1073. }
  1074. user.log.Debugfln("User read chat %s/%s in WhatsApp mobile (last known event: %s/%s)", portal.Key.JID, portal.MXID, message.JID, message.MXID)
  1075. } else {
  1076. message = user.bridge.DB.Message.GetByJID(portal.Key, messageID)
  1077. if message == nil || message.IsFakeMXID() {
  1078. return
  1079. }
  1080. user.log.Debugfln("User read message %s/%s in %s/%s in WhatsApp mobile", message.JID, message.MXID, portal.Key.JID, portal.MXID)
  1081. }
  1082. err := intent.MarkRead(portal.MXID, message.MXID)
  1083. if err != nil {
  1084. user.log.Warnfln("Failed to bridge own read receipt in %s: %v", jid, err)
  1085. }
  1086. }
  1087. func (user *User) HandleCommand(cmd whatsapp.JSONCommand) {
  1088. switch cmd.Type {
  1089. case whatsapp.CommandPicture:
  1090. if strings.HasSuffix(cmd.JID, whatsapp.NewUserSuffix) {
  1091. puppet := user.bridge.GetPuppetByJID(cmd.JID)
  1092. go puppet.UpdateAvatar(user, cmd.ProfilePicInfo)
  1093. } else if user.bridge.Config.Bridge.ChatMetaSync {
  1094. portal := user.GetPortalByJID(cmd.JID)
  1095. go portal.UpdateAvatar(user, cmd.ProfilePicInfo, true)
  1096. }
  1097. case whatsapp.CommandDisconnect:
  1098. if cmd.Kind == "replaced" {
  1099. user.cleanDisconnection = true
  1100. go user.sendMarkdownBridgeAlert("\u26a0 Your WhatsApp connection was closed by the server because you opened another WhatsApp Web client.\n\n" +
  1101. "Use the `reconnect` command to disconnect the other client and resume bridging.")
  1102. } else {
  1103. user.log.Warnln("Unknown kind of disconnect:", string(cmd.Raw))
  1104. go user.sendMarkdownBridgeAlert("\u26a0 Your WhatsApp connection was closed by the server (reason code: %s).\n\n"+
  1105. "Use the `reconnect` command to reconnect.", cmd.Kind)
  1106. }
  1107. }
  1108. }
  1109. func (user *User) HandleChatUpdate(cmd whatsapp.ChatUpdate) {
  1110. if cmd.Command != whatsapp.ChatUpdateCommandAction {
  1111. return
  1112. }
  1113. portal := user.GetPortalByJID(cmd.JID)
  1114. if len(portal.MXID) == 0 {
  1115. if cmd.Data.Action == whatsapp.ChatActionIntroduce || cmd.Data.Action == whatsapp.ChatActionCreate {
  1116. go func() {
  1117. err := portal.CreateMatrixRoom(user)
  1118. if err != nil {
  1119. user.log.Errorln("Failed to create portal room after receiving join event:", err)
  1120. }
  1121. }()
  1122. }
  1123. return
  1124. }
  1125. // These don't come down the message history :(
  1126. switch cmd.Data.Action {
  1127. case whatsapp.ChatActionAddTopic:
  1128. go portal.UpdateTopic(cmd.Data.AddTopic.Topic, cmd.Data.SenderJID, nil, true)
  1129. case whatsapp.ChatActionRemoveTopic:
  1130. go portal.UpdateTopic("", cmd.Data.SenderJID, nil, true)
  1131. case whatsapp.ChatActionRemove:
  1132. // We ignore leaving groups in the message history to avoid accidentally leaving rejoined groups,
  1133. // but if we get a real-time command that says we left, it should be safe to bridge it.
  1134. if !user.bridge.Config.Bridge.ChatMetaSync {
  1135. for _, jid := range cmd.Data.UserChange.JIDs {
  1136. if jid == user.JID {
  1137. go portal.HandleWhatsAppKick(nil, cmd.Data.SenderJID, cmd.Data.UserChange.JIDs)
  1138. break
  1139. }
  1140. }
  1141. }
  1142. }
  1143. if !user.bridge.Config.Bridge.ChatMetaSync {
  1144. // Ignore chat update commands, we're relying on the message history.
  1145. return
  1146. }
  1147. switch cmd.Data.Action {
  1148. case whatsapp.ChatActionNameChange:
  1149. go portal.UpdateName(cmd.Data.NameChange.Name, cmd.Data.SenderJID, nil, true)
  1150. case whatsapp.ChatActionPromote:
  1151. go portal.ChangeAdminStatus(cmd.Data.UserChange.JIDs, true)
  1152. case whatsapp.ChatActionDemote:
  1153. go portal.ChangeAdminStatus(cmd.Data.UserChange.JIDs, false)
  1154. case whatsapp.ChatActionAnnounce:
  1155. go portal.RestrictMessageSending(cmd.Data.Announce)
  1156. case whatsapp.ChatActionRestrict:
  1157. go portal.RestrictMetadataChanges(cmd.Data.Restrict)
  1158. case whatsapp.ChatActionRemove:
  1159. go portal.HandleWhatsAppKick(nil, cmd.Data.SenderJID, cmd.Data.UserChange.JIDs)
  1160. case whatsapp.ChatActionAdd:
  1161. go portal.HandleWhatsAppInvite(cmd.Data.SenderJID, nil, cmd.Data.UserChange.JIDs)
  1162. case whatsapp.ChatActionIntroduce:
  1163. if cmd.Data.SenderJID != "unknown" {
  1164. go portal.Sync(user, whatsapp.Contact{JID: portal.Key.JID})
  1165. }
  1166. }
  1167. }
  1168. func (user *User) HandleConnInfo(info whatsapp.ConnInfo) {
  1169. if user.Session != nil && info.Connected && len(info.ClientToken) > 0 {
  1170. user.log.Debugln("Received new tokens")
  1171. user.Session.ClientToken = info.ClientToken
  1172. user.Session.ServerToken = info.ServerToken
  1173. user.Session.Wid = info.WID
  1174. user.Update()
  1175. }
  1176. if len(info.PushName) > 0 {
  1177. user.pushName = info.PushName
  1178. }
  1179. }
  1180. func (user *User) HandleJSONMessage(message json.RawMessage) {
  1181. if !json.Valid(message) {
  1182. return
  1183. }
  1184. user.log.Debugfln("JSON message: %s", message)
  1185. user.updateLastConnectionIfNecessary()
  1186. }
  1187. func (user *User) NeedsRelaybot(portal *Portal) bool {
  1188. return !user.HasSession() || !user.IsInPortal(portal.Key)
  1189. }