mirror of
https://codeberg.org/Mercury-IM/Smack
synced 2025-01-08 20:47:58 +01:00
Improve ReconnectionManager
Fixes SMACK-778.
This commit is contained in:
parent
1d943aed20
commit
941f29e928
1 changed files with 49 additions and 21 deletions
|
@ -43,7 +43,10 @@ import org.jivesoftware.smack.util.Async;
|
|||
* </ol>
|
||||
*
|
||||
* {@link ReconnectionPolicy#FIXED_DELAY} - The reconnection mechanism will try to reconnect after a fixed delay
|
||||
* independently from the number of reconnection attempts already performed
|
||||
* independently from the number of reconnection attempts already performed.
|
||||
* <p>
|
||||
* Interrupting the reconnection thread will abort the reconnection mechanism.
|
||||
* </p>
|
||||
*
|
||||
* @author Francisco Vives
|
||||
* @author Luca Stucchi
|
||||
|
@ -163,7 +166,7 @@ public final class ReconnectionManager {
|
|||
private ReconnectionManager(AbstractXMPPConnection connection) {
|
||||
weakRefConnection = new WeakReference<AbstractXMPPConnection>(connection);
|
||||
|
||||
reconnectionRunnable = new Thread() {
|
||||
reconnectionRunnable = new Runnable() {
|
||||
|
||||
/**
|
||||
* Holds the current number of reconnection attempts
|
||||
|
@ -211,6 +214,10 @@ public final class ReconnectionManager {
|
|||
if (connection == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Reset attempts to zero since a new reconnection cycle is started once this runs.
|
||||
attempts = 0;
|
||||
|
||||
// The process will try to reconnect until the connection is established or
|
||||
// the user cancel the reconnection process AbstractXMPPConnection.disconnect().
|
||||
while (isReconnectionPossible(connection)) {
|
||||
|
@ -219,7 +226,10 @@ public final class ReconnectionManager {
|
|||
// Sleep until we're ready for the next reconnection attempt. Notify
|
||||
// listeners once per second about how much time remains before the next
|
||||
// reconnection attempt.
|
||||
while (isReconnectionPossible(connection) && remainingSeconds > 0) {
|
||||
while (remainingSeconds > 0) {
|
||||
if (!isReconnectionPossible(connection)) {
|
||||
return;
|
||||
}
|
||||
try {
|
||||
Thread.sleep(1000);
|
||||
remainingSeconds--;
|
||||
|
@ -228,8 +238,9 @@ public final class ReconnectionManager {
|
|||
}
|
||||
}
|
||||
catch (InterruptedException e) {
|
||||
LOGGER.log(Level.FINE, "waiting for reconnection interrupted", e);
|
||||
break;
|
||||
LOGGER.log(Level.FINE, "Reconnection Thread was interrupted, aborting reconnection mechanism", e);
|
||||
// Exit the reconnection thread in case it was interrupted.
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -237,24 +248,18 @@ public final class ReconnectionManager {
|
|||
listener.reconnectingIn(0);
|
||||
}
|
||||
|
||||
if (!isReconnectionPossible(connection)) {
|
||||
return;
|
||||
}
|
||||
// Makes a reconnection attempt
|
||||
try {
|
||||
if (isReconnectionPossible(connection)) {
|
||||
try {
|
||||
connection.connect();
|
||||
} catch (SmackException.AlreadyConnectedException e) {
|
||||
LOGGER.log(Level.FINER, "Connection was already connected on reconnection attempt", e);
|
||||
}
|
||||
try {
|
||||
connection.connect();
|
||||
}
|
||||
// TODO Starting with Smack 4.2, connect() will no
|
||||
// longer login automatically. So change this and the
|
||||
// previous lines to connection.connect().login() in the
|
||||
// 4.2, or any later, branch.
|
||||
if (!connection.isAuthenticated()) {
|
||||
connection.login();
|
||||
catch (SmackException.AlreadyConnectedException e) {
|
||||
LOGGER.log(Level.FINER, "Connection was already connected on reconnection attempt", e);
|
||||
}
|
||||
// Successfully reconnected.
|
||||
attempts = 0;
|
||||
connection.login();
|
||||
}
|
||||
catch (SmackException.AlreadyLoggedInException e) {
|
||||
// This can happen if another thread concurrently triggers a reconnection
|
||||
|
@ -262,12 +267,21 @@ public final class ReconnectionManager {
|
|||
// failure. See also SMACK-725.
|
||||
LOGGER.log(Level.FINER, "Reconnection not required, was already logged in", e);
|
||||
}
|
||||
catch (SmackException | IOException | XMPPException | InterruptedException e) {
|
||||
catch (SmackException | IOException | XMPPException e) {
|
||||
// Fires the failed reconnection notification
|
||||
for (ConnectionListener listener : connection.connectionListeners) {
|
||||
listener.reconnectionFailed(e);
|
||||
}
|
||||
// Failed to reconnect, try again.
|
||||
continue;
|
||||
} catch (InterruptedException e) {
|
||||
LOGGER.log(Level.FINE, "Reconnection Thread was interrupted, aborting reconnection mechanism", e);
|
||||
// Exit the reconnection thread in case it was interrupted.
|
||||
return;
|
||||
}
|
||||
|
||||
// Successfully reconnected .
|
||||
return;
|
||||
}
|
||||
}
|
||||
};
|
||||
|
@ -314,7 +328,7 @@ public final class ReconnectionManager {
|
|||
*
|
||||
* @return true, if the reconnection mechanism is enabled.
|
||||
*/
|
||||
public boolean isAutomaticReconnectEnabled() {
|
||||
public synchronized boolean isAutomaticReconnectEnabled() {
|
||||
return automaticReconnectEnabled;
|
||||
}
|
||||
|
||||
|
@ -348,6 +362,20 @@ public final class ReconnectionManager {
|
|||
"Smack Reconnection Manager (" + connection.getConnectionCounter() + ')');
|
||||
}
|
||||
|
||||
/**
|
||||
* Abort a possibly running reconnection mechanism.
|
||||
*
|
||||
* @since 4.2.2
|
||||
*/
|
||||
public synchronized void abortPossiblyRunningReconnection() {
|
||||
if (reconnectionThread == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
reconnectionThread.interrupt();
|
||||
reconnectionThread = null;
|
||||
}
|
||||
|
||||
private final ConnectionListener connectionListener = new AbstractConnectionListener() {
|
||||
|
||||
@Override
|
||||
|
|
Loading…
Reference in a new issue