Handle 400 init response and add ping command. Fixes #87

This commit is contained in:
Tulir Asokan 2019-08-24 22:39:12 +03:00
parent b9ce4d7ab4
commit 44a0fc65b8
4 changed files with 54 additions and 30 deletions

View File

@ -90,6 +90,8 @@ func (handler *CommandHandler) Handle(roomID types.MatrixRoomID, user *User, mes
handler.CommandReconnect(ce) handler.CommandReconnect(ce)
case "disconnect": case "disconnect":
handler.CommandDisconnect(ce) handler.CommandDisconnect(ce)
case "ping":
handler.CommandPing(ce)
case "delete-connection": case "delete-connection":
handler.CommandDeleteConnection(ce) handler.CommandDeleteConnection(ce)
case "delete-session": case "delete-session":
@ -98,11 +100,13 @@ func (handler *CommandHandler) Handle(roomID types.MatrixRoomID, user *User, mes
handler.CommandDeletePortal(ce) handler.CommandDeletePortal(ce)
case "delete-all-portals": case "delete-all-portals":
handler.CommandDeleteAllPortals(ce) handler.CommandDeleteAllPortals(ce)
case "dev-test":
handler.CommandDevTest(ce)
case "login-matrix", "logout", "sync", "list", "open", "pm": case "login-matrix", "logout", "sync", "list", "open", "pm":
if ce.User.Conn == nil { if ce.User.Conn == nil {
ce.Reply("You are not logged in. Use the `login` command to log into WhatsApp.") ce.Reply("You are not logged in. Use the `login` command to log into WhatsApp.")
return return
} else if !ce.User.Connected { } else if !ce.User.IsConnected() {
ce.Reply("You are not connected to WhatsApp. Use the `reconnect` command to reconnect.") ce.Reply("You are not connected to WhatsApp. Use the `reconnect` command to reconnect.")
return return
} }
@ -126,6 +130,10 @@ func (handler *CommandHandler) Handle(roomID types.MatrixRoomID, user *User, mes
} }
} }
func (handler *CommandHandler) CommandDevTest(ce *CommandEvent) {
}
const cmdLoginHelp = `login - Authenticate this Bridge as WhatsApp Web Client` const cmdLoginHelp = `login - Authenticate this Bridge as WhatsApp Web Client`
// CommandLogin handles login command // CommandLogin handles login command
@ -157,7 +165,6 @@ func (handler *CommandHandler) CommandLogout(ce *CommandEvent) {
if err != nil { if err != nil {
ce.User.log.Warnln("Error while disconnecting after logout:", err) ce.User.log.Warnln("Error while disconnecting after logout:", err)
} }
ce.User.Connected = false
ce.User.Conn.RemoveHandlers() ce.User.Conn.RemoveHandlers()
ce.User.Conn = nil ce.User.Conn = nil
ce.User.SetSession(nil) ce.User.SetSession(nil)
@ -167,12 +174,11 @@ func (handler *CommandHandler) CommandLogout(ce *CommandEvent) {
const cmdDeleteSessionHelp = `delete-session - Delete session information and disconnect from WhatsApp without sending a logout request` const cmdDeleteSessionHelp = `delete-session - Delete session information and disconnect from WhatsApp without sending a logout request`
func (handler *CommandHandler) CommandDeleteSession(ce *CommandEvent) { func (handler *CommandHandler) CommandDeleteSession(ce *CommandEvent) {
if ce.User.Session == nil && !ce.User.Connected && ce.User.Conn == nil { if ce.User.Session == nil && ce.User.Conn == nil {
ce.Reply("Nothing to purge: no session information stored and no active connection.") ce.Reply("Nothing to purge: no session information stored and no active connection.")
return return
} }
ce.User.SetSession(nil) ce.User.SetSession(nil)
ce.User.Connected = false
if ce.User.Conn != nil { if ce.User.Conn != nil {
_, _ = ce.User.Conn.Disconnect() _, _ = ce.User.Conn.Disconnect()
ce.User.Conn.RemoveHandlers() ce.User.Conn.RemoveHandlers()
@ -213,13 +219,7 @@ func (handler *CommandHandler) CommandReconnect(ce *CommandEvent) {
if err != nil { if err != nil {
ce.User.log.Warnln("Error while reconnecting:", err) ce.User.log.Warnln("Error while reconnecting:", err)
if err == whatsapp.ErrAlreadyLoggedIn { if err == whatsapp.ErrAlreadyLoggedIn {
if ce.User.Connected { ce.Reply("You were already connected.")
ce.Reply("You were already connected.")
} else {
ce.User.Connected = true
ce.User.ConnectionErrors = 0
ce.Reply("You were already connected, but the bridge hadn't noticed. Fixed that now.")
}
} else if err.Error() == "restore session connection timed out" { } else if err.Error() == "restore session connection timed out" {
ce.Reply("Reconnection timed out. Is WhatsApp on your phone reachable?") ce.Reply("Reconnection timed out. Is WhatsApp on your phone reachable?")
} else { } else {
@ -227,7 +227,6 @@ func (handler *CommandHandler) CommandReconnect(ce *CommandEvent) {
} }
return return
} }
ce.User.Connected = true
ce.User.ConnectionErrors = 0 ce.User.ConnectionErrors = 0
ce.Reply("Reconnected successfully.") ce.Reply("Reconnected successfully.")
ce.User.PostLogin() ce.User.PostLogin()
@ -242,7 +241,6 @@ func (handler *CommandHandler) CommandDeleteConnection(ce *CommandEvent) {
if err == nil && len(sess.Wid) > 0 { if err == nil && len(sess.Wid) > 0 {
ce.User.SetSession(&sess) ce.User.SetSession(&sess)
} }
ce.User.Connected = false
ce.User.Conn.RemoveHandlers() ce.User.Conn.RemoveHandlers()
ce.User.Conn = nil ce.User.Conn = nil
ce.Reply("Successfully disconnected. Use the `reconnect` command to reconnect.") ce.Reply("Successfully disconnected. Use the `reconnect` command to reconnect.")
@ -256,7 +254,6 @@ func (handler *CommandHandler) CommandDisconnect(ce *CommandEvent) {
return return
} }
sess, err := ce.User.Conn.Disconnect() sess, err := ce.User.Conn.Disconnect()
ce.User.Connected = false
if err == whatsapp.ErrNotConnected { if err == whatsapp.ErrNotConnected {
ce.Reply("You were not connected.") ce.Reply("You were not connected.")
return return
@ -269,6 +266,23 @@ func (handler *CommandHandler) CommandDisconnect(ce *CommandEvent) {
ce.Reply("Successfully disconnected. Use the `reconnect` command to reconnect.") ce.Reply("Successfully disconnected. Use the `reconnect` command to reconnect.")
} }
const cmdPingHelp = `ping - Check your connection to WhatsApp.`
func (handler *CommandHandler) CommandPing(ce *CommandEvent) {
if ce.User.Conn == nil {
ce.Reply("You don't have a WhatsApp connection.")
return
}
ok, err := ce.User.Conn.AdminTest()
if err != nil {
ce.Reply("Connection not OK: %v", err)
} else if !ok {
ce.Reply("Connection not OK, but no error received")
} else {
ce.Reply("Connection to WhatsApp OK")
}
}
const cmdHelpHelp = `help - Prints this help` const cmdHelpHelp = `help - Prints this help`
// CommandHelp handles help command // CommandHelp handles help command
@ -285,6 +299,9 @@ func (handler *CommandHandler) CommandHelp(ce *CommandEvent) {
cmdPrefix + cmdDeleteSessionHelp, cmdPrefix + cmdDeleteSessionHelp,
cmdPrefix + cmdReconnectHelp, cmdPrefix + cmdReconnectHelp,
cmdPrefix + cmdDisconnectHelp, cmdPrefix + cmdDisconnectHelp,
cmdPrefix + cmdPingHelp,
cmdPrefix + cmdLoginMatrixHelp,
cmdPrefix + cmdLogoutMatrixHelp,
cmdPrefix + cmdSyncHelp, cmdPrefix + cmdSyncHelp,
cmdPrefix + cmdListHelp, cmdPrefix + cmdListHelp,
cmdPrefix + cmdOpenHelp, cmdPrefix + cmdOpenHelp,

View File

@ -136,8 +136,8 @@ func (puppet *Puppet) stopSyncing() {
} }
func (puppet *Puppet) ProcessResponse(resp *mautrix.RespSync, since string) error { func (puppet *Puppet) ProcessResponse(resp *mautrix.RespSync, since string) error {
if !puppet.customUser.Connected { if !puppet.customUser.IsConnected() {
puppet.log.Warnln("Skipping sync processing: custom user not connected to whatsapp") puppet.log.Debugln("Skipping sync processing: custom user not connected to whatsapp")
return nil return nil
} }
for roomID, events := range resp.Rooms.Join { for roomID, events := range resp.Rooms.Join {

View File

@ -118,7 +118,7 @@ func (mx *MatrixHandler) HandleMembership(evt *mautrix.Event) {
} }
user := mx.bridge.GetUserByMXID(types.MatrixUserID(evt.Sender)) user := mx.bridge.GetUserByMXID(types.MatrixUserID(evt.Sender))
if user == nil || !user.Whitelisted || !user.IsLoggedIn() { if user == nil || !user.Whitelisted || !user.IsConnected() {
return return
} }
@ -135,7 +135,7 @@ func (mx *MatrixHandler) HandleMembership(evt *mautrix.Event) {
func (mx *MatrixHandler) HandleRoomMetadata(evt *mautrix.Event) { func (mx *MatrixHandler) HandleRoomMetadata(evt *mautrix.Event) {
user := mx.bridge.GetUserByMXID(types.MatrixUserID(evt.Sender)) user := mx.bridge.GetUserByMXID(types.MatrixUserID(evt.Sender))
if user == nil || !user.Whitelisted || !user.IsLoggedIn() || !user.Connected { if user == nil || !user.Whitelisted || !user.IsConnected() {
return return
} }
@ -190,9 +190,9 @@ func (mx *MatrixHandler) HandleMessage(evt *mautrix.Event) {
} }
} }
if !user.IsLoggedIn() { if !user.HasSession() {
return return
} else if !user.Connected { } else if !user.IsConnected() {
msg := format.RenderMarkdown(fmt.Sprintf("\u26a0 You are not connected to WhatsApp, so your message was not bridged. " + msg := format.RenderMarkdown(fmt.Sprintf("\u26a0 You are not connected to WhatsApp, so your message was not bridged. " +
"Use `%s reconnect` to reconnect.", mx.bridge.Config.Bridge.CommandPrefix)) "Use `%s reconnect` to reconnect.", mx.bridge.Config.Bridge.CommandPrefix))
msg.MsgType = mautrix.MsgNotice msg.MsgType = mautrix.MsgNotice
@ -218,9 +218,9 @@ func (mx *MatrixHandler) HandleRedaction(evt *mautrix.Event) {
return return
} }
if !user.IsLoggedIn() { if !user.HasSession() {
return return
} else if !user.Connected { } else if !user.IsConnected() {
msg := format.RenderMarkdown(fmt.Sprintf("[%[1]s](https://matrix.to/#/%[1]s): \u26a0 " + 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. " + "You are not connected to WhatsApp, so your redaction was not bridged. " +
"Use `%[2]s reconnect` to reconnect.", user.MXID, mx.bridge.Config.Bridge.CommandPrefix)) "Use `%[2]s reconnect` to reconnect.", user.MXID, mx.bridge.Config.Bridge.CommandPrefix))

23
user.go
View File

@ -49,7 +49,6 @@ type User struct {
Admin bool Admin bool
Whitelisted bool Whitelisted bool
Connected bool
ConnectionErrors int ConnectionErrors int
CommunityID string CommunityID string
@ -208,7 +207,6 @@ func (user *User) RestoreSession() bool {
_, _ = user.bridge.Bot.SendMessageEvent(user.ManagementRoom, mautrix.EventMessage, msg) _, _ = user.bridge.Bot.SendMessageEvent(user.ManagementRoom, mautrix.EventMessage, msg)
return false return false
} }
user.Connected = true
user.ConnectionErrors = 0 user.ConnectionErrors = 0
user.SetSession(&sess) user.SetSession(&sess)
user.log.Debugln("Session restored successfully") user.log.Debugln("Session restored successfully")
@ -217,8 +215,12 @@ func (user *User) RestoreSession() bool {
return true return true
} }
func (user *User) IsLoggedIn() bool { func (user *User) HasSession() bool {
return user.Session != nil || user.Conn != nil return user.Session != nil
}
func (user *User) IsConnected() bool {
return user.Conn != nil && user.Conn.IsConnected() && user.Conn.IsLoggedIn()
} }
func (user *User) loginQrChannel(ce *CommandEvent, qrChan <-chan string, eventIDChan chan<- string) { func (user *User) loginQrChannel(ce *CommandEvent, qrChan <-chan string, eventIDChan chan<- string) {
@ -310,7 +312,6 @@ func (user *User) Login(ce *CommandEvent) {
_, _ = ce.Bot.SendMessageEvent(ce.RoomID, mautrix.EventMessage, &msg) _, _ = ce.Bot.SendMessageEvent(ce.RoomID, mautrix.EventMessage, &msg)
return return
} }
user.Connected = true
user.ConnectionErrors = 0 user.ConnectionErrors = 0
user.JID = strings.Replace(user.Conn.Info.Wid, whatsappExt.OldUserSuffix, whatsappExt.NewUserSuffix, 1) user.JID = strings.Replace(user.Conn.Info.Wid, whatsappExt.OldUserSuffix, whatsappExt.NewUserSuffix, 1)
user.SetSession(&session) user.SetSession(&session)
@ -432,7 +433,6 @@ func (user *User) HandleError(err error) {
user.log.Errorln("WhatsApp error:", err) user.log.Errorln("WhatsApp error:", err)
} }
if closed, ok := err.(*whatsapp.ErrConnectionClosed); ok { if closed, ok := err.(*whatsapp.ErrConnectionClosed); ok {
user.Connected = false
if closed.Code == 1000 && user.cleanDisconnection { if closed.Code == 1000 && user.cleanDisconnection {
user.cleanDisconnection = false user.cleanDisconnection = false
user.log.Infoln("Clean disconnection by server") user.log.Infoln("Clean disconnection by server")
@ -440,7 +440,6 @@ func (user *User) HandleError(err error) {
} }
go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection was closed with websocket status code %d", closed.Code)) go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection was closed with websocket status code %d", closed.Code))
} else if failed, ok := err.(*whatsapp.ErrConnectionFailed); ok { } else if failed, ok := err.(*whatsapp.ErrConnectionFailed); ok {
user.Connected = false
user.ConnectionErrors++ user.ConnectionErrors++
go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection failed: %v", failed.Err)) go user.tryReconnect(fmt.Sprintf("Your WhatsApp connection failed: %v", failed.Err))
} }
@ -470,10 +469,18 @@ func (user *User) tryReconnect(msg string) {
err := user.Conn.Restore() err := user.Conn.Restore()
if err == nil { if err == nil {
user.ConnectionErrors = 0 user.ConnectionErrors = 0
user.Connected = true
_, _ = user.bridge.Bot.SendNotice(user.ManagementRoom, "Reconnected successfully") _, _ = user.bridge.Bot.SendNotice(user.ManagementRoom, "Reconnected successfully")
user.PostLogin() user.PostLogin()
return return
} else if err.Error() == "init responded with 400" {
user.log.Infoln("Got init 400 error when trying to reconnect, resetting connection...")
sess, err := user.Conn.Disconnect()
if err != nil {
user.log.Debugln("Error while disconnecting for connection reset:", err)
}
if len(sess.Wid) > 0 {
user.SetSession(&sess)
}
} }
user.log.Errorln("Error while trying to reconnect after disconnection:", err) user.log.Errorln("Error while trying to reconnect after disconnection:", err)
tries++ tries++