Mercury-IM/domain/src/main/java/org/mercury_im/messenger/core/connection/MercuryConnection.java

185 lines
7.1 KiB
Java

package org.mercury_im.messenger.core.connection;
import org.jivesoftware.smack.AbstractXMPPConnection;
import org.jivesoftware.smack.ConnectionListener;
import org.jivesoftware.smack.ReconnectionListener;
import org.jivesoftware.smack.ReconnectionManager;
import org.jivesoftware.smack.SmackException;
import org.jivesoftware.smack.XMPPConnection;
import org.jivesoftware.smack.XMPPException;
import org.jivesoftware.smack.sasl.SASLErrorException;
import org.mercury_im.messenger.core.connection.exception.InvalidCredentialsException;
import org.mercury_im.messenger.core.connection.exception.ServerUnreachableException;
import org.mercury_im.messenger.core.connection.state.ConnectionState;
import org.mercury_im.messenger.core.connection.state.ConnectivityState;
import org.mercury_im.messenger.entity.Account;
import java.io.IOException;
import java.util.UUID;
import java.util.concurrent.atomic.AtomicReference;
import java.util.logging.Level;
import java.util.logging.Logger;
import io.reactivex.Completable;
import io.reactivex.Observable;
import io.reactivex.subjects.BehaviorSubject;
import lombok.Getter;
public class MercuryConnection {
private static final Logger LOGGER = Logger.getLogger(MercuryConnection.class.getName());
@Getter
private XMPPConnection connection;
@Getter
private final Account account;
private final MercuryConnectionListener connectionListener;
public MercuryConnection(XMPPConnection connection, Account account) {
this.connection = connection;
this.account = account;
if (connection.isConnected()) {
throw new IllegalStateException("Connection" + account.getJid() + " MUST NOT be connected at this point.");
}
connectionListener = new MercuryConnectionListener(account.getId(), this);
connection.addConnectionListener(connectionListener);
initialConnectionSetup();
}
public UUID getAccountId() {
return getAccount().getId();
}
public Observable<ConnectionState> observeConnection() {
return connectionListener.connectionState;
}
public Completable connect() {
return Completable.fromAction(this::doConnect)
.doOnError(error -> LOGGER.log(Level.WARNING, "connect(): Connection error for account " + account, error));
}
private synchronized void doConnect() throws ServerUnreachableException {
ConnectivityState connectivity = connectionListener.connectivity.get();
if (connectivity != ConnectivityState.disconnected) {
LOGGER.log(Level.WARNING, "doConnect(): Connection " + account.getJid() + " is not disconnected: " + connectivity);
return;
}
AbstractXMPPConnection connection = (AbstractXMPPConnection) getConnection();
try {
connection.connect();
} catch (SmackException.EndpointConnectionException e) {
connection.disconnect();
throw new ServerUnreachableException("doConnect(): Cannot connect to server " + connection.getXMPPServiceDomain().asUnescapedString(), e);
} catch (IOException | InterruptedException | XMPPException | SmackException e) {
throw new AssertionError("Unexpected exception.", e);
}
LOGGER.log(Level.INFO, "Connected!");
}
public Completable login() {
return Completable.fromAction(this::doLogin)
.doOnError(error -> LOGGER.log(Level.WARNING, "Login error for account " + account, error));
}
private synchronized void doLogin() throws InvalidCredentialsException {
ConnectivityState connectivity = connectionListener.connectivity.get();
if (connectivity != ConnectivityState.connected) {
LOGGER.log(Level.WARNING, "doLogin(): Connection " + account.getJid() + " is not connected: " + connectivity);
}
try {
((AbstractXMPPConnection) getConnection()).login();
} catch (SASLErrorException e) {
throw new InvalidCredentialsException("Credentials of account " + account.getId() + " are invalid.", e);
} catch (InterruptedException | XMPPException | SmackException | IOException e) {
throw new AssertionError("Unexpected exception.", e);
}
}
public Completable shutdown() {
return Completable.fromAction(this::doShutdown)
.doOnError(error -> LOGGER.log(Level.WARNING, "Shutdown error for account " + account.getJid(), error));
}
public synchronized void doShutdown() {
if (connection.isConnected()) {
((AbstractXMPPConnection) getConnection()).disconnect();
} else {
((AbstractXMPPConnection) getConnection()).instantShutdown();
}
}
private void initialConnectionSetup() {
ReconnectionManager.getInstanceFor((AbstractXMPPConnection) getConnection())
.addReconnectionListener(new ReconnectionListener() {
@Override
public void reconnectingIn(int seconds) {
LOGGER.log(Level.FINER, "Reconnecting connection " + getAccount().getAddress() + " in " + seconds + " seconds.");
}
@Override
public void reconnectionFailed(Exception e) {
LOGGER.log(Level.WARNING, "Reconnection of connection " + getAccount().getAddress() + " failed.", e);
}
});
}
private class MercuryConnectionListener implements ConnectionListener {
private final AtomicReference<ConnectivityState> connectivity;
private final BehaviorSubject<ConnectionState> connectionState;
public MercuryConnectionListener(UUID accountId, MercuryConnection connection) {
this.connectionState = BehaviorSubject.createDefault(new ConnectionState(accountId, connection));
this.connectivity = new AtomicReference<>(ConnectivityState.disconnected);
}
@Override
public void connecting(XMPPConnection connection) {
changeConnectivity(ConnectivityState.connecting);
}
@Override
public void connected(XMPPConnection connection) {
changeConnectivity(ConnectivityState.connected);
}
@Override
public void authenticated(XMPPConnection connection, boolean resumed) {
connectivity.set(ConnectivityState.authenticated);
ConnectionState state = connectionState.getValue()
.withConnectivity(ConnectivityState.authenticated)
.withResumed(resumed);
connectionState.onNext(state);
}
@Override
public void connectionClosed() {
changeConnectivity(ConnectivityState.disconnected);
}
@Override
public void connectionClosedOnError(Exception e) {
changeConnectivity(ConnectivityState.disconnectedOnError);
}
private void changeConnectivity(ConnectivityState newConnectivity) {
connectivity.set(newConnectivity);
ConnectionState state = connectionState.getValue()
.withConnectivity(newConnectivity);
connectionState.onNext(state);
}
}
}